2021-10-29 21:25:12 +01:00
|
|
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
|
|
#include "Luau/TxnLog.h"
|
|
|
|
|
2023-05-12 13:15:01 +01:00
|
|
|
#include "Luau/Scope.h"
|
2022-02-03 23:09:37 +00:00
|
|
|
#include "Luau/ToString.h"
|
2022-12-09 18:07:25 +00:00
|
|
|
#include "Luau/TypeArena.h"
|
2021-10-29 21:25:12 +01:00
|
|
|
#include "Luau/TypePack.h"
|
|
|
|
|
|
|
|
#include <algorithm>
|
2022-01-06 22:10:07 +00:00
|
|
|
#include <stdexcept>
|
|
|
|
|
2023-05-12 13:15:01 +01:00
|
|
|
LUAU_FASTFLAG(DebugLuauDeferredConstraintResolution)
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
namespace Luau
|
|
|
|
{
|
|
|
|
|
2022-02-03 23:09:37 +00:00
|
|
|
const std::string nullPendingResult = "<nullptr>";
|
|
|
|
|
|
|
|
std::string toString(PendingType* pending)
|
|
|
|
{
|
|
|
|
if (pending == nullptr)
|
|
|
|
return nullPendingResult;
|
|
|
|
|
|
|
|
return toString(pending->pending);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string dump(PendingType* pending)
|
|
|
|
{
|
|
|
|
if (pending == nullptr)
|
|
|
|
{
|
|
|
|
printf("%s\n", nullPendingResult.c_str());
|
|
|
|
return nullPendingResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
ToStringOptions opts;
|
|
|
|
opts.exhaustive = true;
|
|
|
|
opts.functionTypeArguments = true;
|
|
|
|
std::string result = toString(pending->pending, opts);
|
|
|
|
printf("%s\n", result.c_str());
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string toString(PendingTypePack* pending)
|
|
|
|
{
|
|
|
|
if (pending == nullptr)
|
|
|
|
return nullPendingResult;
|
|
|
|
|
|
|
|
return toString(pending->pending);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string dump(PendingTypePack* pending)
|
|
|
|
{
|
|
|
|
if (pending == nullptr)
|
|
|
|
{
|
|
|
|
printf("%s\n", nullPendingResult.c_str());
|
|
|
|
return nullPendingResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
ToStringOptions opts;
|
|
|
|
opts.exhaustive = true;
|
|
|
|
opts.functionTypeArguments = true;
|
|
|
|
std::string result = toString(pending->pending, opts);
|
|
|
|
printf("%s\n", result.c_str());
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2022-01-06 22:10:07 +00:00
|
|
|
static const TxnLog emptyLog;
|
|
|
|
|
|
|
|
const TxnLog* TxnLog::empty()
|
|
|
|
{
|
|
|
|
return &emptyLog;
|
|
|
|
}
|
|
|
|
|
|
|
|
void TxnLog::concat(TxnLog rhs)
|
|
|
|
{
|
|
|
|
for (auto& [ty, rep] : rhs.typeVarChanges)
|
2023-05-12 13:15:01 +01:00
|
|
|
{
|
|
|
|
if (rep->dead)
|
|
|
|
continue;
|
2022-01-06 22:10:07 +00:00
|
|
|
typeVarChanges[ty] = std::move(rep);
|
2023-05-12 13:15:01 +01:00
|
|
|
}
|
2022-01-06 22:10:07 +00:00
|
|
|
|
|
|
|
for (auto& [tp, rep] : rhs.typePackChanges)
|
|
|
|
typePackChanges[tp] = std::move(rep);
|
|
|
|
}
|
|
|
|
|
2022-12-02 10:46:05 +00:00
|
|
|
void TxnLog::concatAsIntersections(TxnLog rhs, NotNull<TypeArena> arena)
|
|
|
|
{
|
|
|
|
for (auto& [ty, rightRep] : rhs.typeVarChanges)
|
|
|
|
{
|
2023-05-12 13:15:01 +01:00
|
|
|
if (rightRep->dead)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (auto leftRep = typeVarChanges.find(ty); leftRep && !(*leftRep)->dead)
|
2022-12-02 10:46:05 +00:00
|
|
|
{
|
|
|
|
TypeId leftTy = arena->addType((*leftRep)->pending);
|
|
|
|
TypeId rightTy = arena->addType(rightRep->pending);
|
2023-01-03 17:33:19 +00:00
|
|
|
typeVarChanges[ty]->pending.ty = IntersectionType{{leftTy, rightTy}};
|
2022-12-02 10:46:05 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
typeVarChanges[ty] = std::move(rightRep);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto& [tp, rep] : rhs.typePackChanges)
|
|
|
|
typePackChanges[tp] = std::move(rep);
|
|
|
|
}
|
|
|
|
|
|
|
|
void TxnLog::concatAsUnion(TxnLog rhs, NotNull<TypeArena> arena)
|
|
|
|
{
|
2023-05-12 13:15:01 +01:00
|
|
|
if (FFlag::DebugLuauDeferredConstraintResolution)
|
2022-12-02 10:46:05 +00:00
|
|
|
{
|
2023-05-12 13:15:01 +01:00
|
|
|
/*
|
|
|
|
* Check for cycles.
|
|
|
|
*
|
|
|
|
* We must not combine a log entry that binds 'a to 'b with a log that
|
|
|
|
* binds 'b to 'a.
|
|
|
|
*
|
|
|
|
* Of the two, identify the one with the 'bigger' scope and eliminate the
|
|
|
|
* entry that rebinds it.
|
|
|
|
*/
|
|
|
|
for (const auto& [rightTy, rightRep] : rhs.typeVarChanges)
|
2022-12-02 10:46:05 +00:00
|
|
|
{
|
2023-05-12 13:15:01 +01:00
|
|
|
if (rightRep->dead)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// We explicitly use get_if here because we do not wish to do anything
|
|
|
|
// if the uncommitted type is already bound to something else.
|
|
|
|
const FreeType* rf = get_if<FreeType>(&rightTy->ty);
|
|
|
|
if (!rf)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
const BoundType* rb = Luau::get<BoundType>(&rightRep->pending);
|
|
|
|
if (!rb)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
const TypeId leftTy = rb->boundTo;
|
|
|
|
const FreeType* lf = get_if<FreeType>(&leftTy->ty);
|
|
|
|
if (!lf)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
auto leftRep = typeVarChanges.find(leftTy);
|
|
|
|
if (!leftRep)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if ((*leftRep)->dead)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
const BoundType* lb = Luau::get<BoundType>(&(*leftRep)->pending);
|
|
|
|
if (!lb)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (lb->boundTo == rightTy)
|
|
|
|
{
|
|
|
|
// leftTy has been bound to rightTy, but rightTy has also been bound
|
|
|
|
// to leftTy. We find the one that belongs to the more deeply nested
|
|
|
|
// scope and remove it from the log.
|
|
|
|
const bool discardLeft = useScopes ? subsumes(lf->scope, rf->scope) : lf->level.subsumes(rf->level);
|
|
|
|
|
|
|
|
if (discardLeft)
|
|
|
|
(*leftRep)->dead = true;
|
|
|
|
else
|
|
|
|
rightRep->dead = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto& [ty, rightRep] : rhs.typeVarChanges)
|
|
|
|
{
|
|
|
|
if (rightRep->dead)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (auto leftRep = typeVarChanges.find(ty); leftRep && !(*leftRep)->dead)
|
|
|
|
{
|
|
|
|
TypeId leftTy = arena->addType((*leftRep)->pending);
|
|
|
|
TypeId rightTy = arena->addType(rightRep->pending);
|
|
|
|
|
|
|
|
if (follow(leftTy) == follow(rightTy))
|
|
|
|
typeVarChanges[ty] = std::move(rightRep);
|
|
|
|
else
|
|
|
|
typeVarChanges[ty]->pending.ty = UnionType{{leftTy, rightTy}};
|
|
|
|
}
|
|
|
|
else
|
|
|
|
typeVarChanges[ty] = std::move(rightRep);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for (auto& [ty, rightRep] : rhs.typeVarChanges)
|
|
|
|
{
|
|
|
|
if (auto leftRep = typeVarChanges.find(ty))
|
|
|
|
{
|
|
|
|
TypeId leftTy = arena->addType((*leftRep)->pending);
|
|
|
|
TypeId rightTy = arena->addType(rightRep->pending);
|
|
|
|
typeVarChanges[ty]->pending.ty = UnionType{{leftTy, rightTy}};
|
|
|
|
}
|
|
|
|
else
|
|
|
|
typeVarChanges[ty] = std::move(rightRep);
|
2022-12-02 10:46:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto& [tp, rep] : rhs.typePackChanges)
|
|
|
|
typePackChanges[tp] = std::move(rep);
|
|
|
|
}
|
|
|
|
|
2022-01-06 22:10:07 +00:00
|
|
|
void TxnLog::commit()
|
|
|
|
{
|
2022-05-06 00:52:48 +01:00
|
|
|
for (auto& [ty, rep] : typeVarChanges)
|
2023-05-12 13:15:01 +01:00
|
|
|
{
|
|
|
|
if (!rep->dead)
|
|
|
|
asMutable(ty)->reassign(rep.get()->pending);
|
|
|
|
}
|
2022-01-06 22:10:07 +00:00
|
|
|
|
2022-05-06 00:52:48 +01:00
|
|
|
for (auto& [tp, rep] : typePackChanges)
|
2022-07-08 02:05:31 +01:00
|
|
|
asMutable(tp)->reassign(rep.get()->pending);
|
2022-01-06 22:10:07 +00:00
|
|
|
|
|
|
|
clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void TxnLog::clear()
|
|
|
|
{
|
|
|
|
typeVarChanges.clear();
|
|
|
|
typePackChanges.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
TxnLog TxnLog::inverse()
|
|
|
|
{
|
|
|
|
TxnLog inversed(sharedSeen);
|
|
|
|
|
|
|
|
for (auto& [ty, _rep] : typeVarChanges)
|
2023-05-12 13:15:01 +01:00
|
|
|
{
|
|
|
|
if (!_rep->dead)
|
|
|
|
inversed.typeVarChanges[ty] = std::make_unique<PendingType>(*ty);
|
|
|
|
}
|
2022-01-06 22:10:07 +00:00
|
|
|
|
|
|
|
for (auto& [tp, _rep] : typePackChanges)
|
|
|
|
inversed.typePackChanges[tp] = std::make_unique<PendingTypePack>(*tp);
|
|
|
|
|
|
|
|
return inversed;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool TxnLog::haveSeen(TypeId lhs, TypeId rhs) const
|
2022-03-04 16:19:20 +00:00
|
|
|
{
|
|
|
|
return haveSeen((TypeOrPackId)lhs, (TypeOrPackId)rhs);
|
|
|
|
}
|
|
|
|
|
|
|
|
void TxnLog::pushSeen(TypeId lhs, TypeId rhs)
|
|
|
|
{
|
|
|
|
pushSeen((TypeOrPackId)lhs, (TypeOrPackId)rhs);
|
|
|
|
}
|
|
|
|
|
|
|
|
void TxnLog::popSeen(TypeId lhs, TypeId rhs)
|
|
|
|
{
|
|
|
|
popSeen((TypeOrPackId)lhs, (TypeOrPackId)rhs);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool TxnLog::haveSeen(TypePackId lhs, TypePackId rhs) const
|
|
|
|
{
|
|
|
|
return haveSeen((TypeOrPackId)lhs, (TypeOrPackId)rhs);
|
|
|
|
}
|
|
|
|
|
|
|
|
void TxnLog::pushSeen(TypePackId lhs, TypePackId rhs)
|
|
|
|
{
|
|
|
|
pushSeen((TypeOrPackId)lhs, (TypeOrPackId)rhs);
|
|
|
|
}
|
|
|
|
|
|
|
|
void TxnLog::popSeen(TypePackId lhs, TypePackId rhs)
|
|
|
|
{
|
|
|
|
popSeen((TypeOrPackId)lhs, (TypeOrPackId)rhs);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool TxnLog::haveSeen(TypeOrPackId lhs, TypeOrPackId rhs) const
|
2022-01-06 22:10:07 +00:00
|
|
|
{
|
2022-05-20 00:46:52 +01:00
|
|
|
const std::pair<TypeOrPackId, TypeOrPackId> sortedPair = (lhs > rhs) ? std::make_pair(lhs, rhs) : std::make_pair(rhs, lhs);
|
|
|
|
if (sharedSeen->end() != std::find(sharedSeen->begin(), sharedSeen->end(), sortedPair))
|
2022-04-14 22:57:15 +01:00
|
|
|
{
|
2022-05-20 00:46:52 +01:00
|
|
|
return true;
|
2022-04-14 22:57:15 +01:00
|
|
|
}
|
2022-05-20 00:46:52 +01:00
|
|
|
|
|
|
|
return false;
|
2022-01-06 22:10:07 +00:00
|
|
|
}
|
|
|
|
|
2022-03-04 16:19:20 +00:00
|
|
|
void TxnLog::pushSeen(TypeOrPackId lhs, TypeOrPackId rhs)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-04 16:19:20 +00:00
|
|
|
const std::pair<TypeOrPackId, TypeOrPackId> sortedPair = (lhs > rhs) ? std::make_pair(lhs, rhs) : std::make_pair(rhs, lhs);
|
2021-12-02 23:20:08 +00:00
|
|
|
sharedSeen->push_back(sortedPair);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-03-04 16:19:20 +00:00
|
|
|
void TxnLog::popSeen(TypeOrPackId lhs, TypeOrPackId rhs)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-04 16:19:20 +00:00
|
|
|
const std::pair<TypeOrPackId, TypeOrPackId> sortedPair = (lhs > rhs) ? std::make_pair(lhs, rhs) : std::make_pair(rhs, lhs);
|
2021-12-02 23:20:08 +00:00
|
|
|
LUAU_ASSERT(sortedPair == sharedSeen->back());
|
|
|
|
sharedSeen->pop_back();
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-01-06 22:10:07 +00:00
|
|
|
PendingType* TxnLog::queue(TypeId ty)
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(!ty->persistent);
|
|
|
|
|
|
|
|
// Explicitly don't look in ancestors. If we have discovered something new
|
|
|
|
// about this type, we don't want to mutate the parent's state.
|
|
|
|
auto& pending = typeVarChanges[ty];
|
2023-05-12 13:15:01 +01:00
|
|
|
if (!pending || (*pending).dead)
|
2022-06-17 01:52:23 +01:00
|
|
|
{
|
2022-01-06 22:10:07 +00:00
|
|
|
pending = std::make_unique<PendingType>(*ty);
|
2022-07-08 02:05:31 +01:00
|
|
|
pending->pending.owningArena = nullptr;
|
2022-06-17 01:52:23 +01:00
|
|
|
}
|
|
|
|
|
2022-01-06 22:10:07 +00:00
|
|
|
return pending.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
PendingTypePack* TxnLog::queue(TypePackId tp)
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(!tp->persistent);
|
|
|
|
|
|
|
|
// Explicitly don't look in ancestors. If we have discovered something new
|
|
|
|
// about this type, we don't want to mutate the parent's state.
|
|
|
|
auto& pending = typePackChanges[tp];
|
|
|
|
if (!pending)
|
2022-06-17 01:52:23 +01:00
|
|
|
{
|
2022-01-06 22:10:07 +00:00
|
|
|
pending = std::make_unique<PendingTypePack>(*tp);
|
2022-07-08 02:05:31 +01:00
|
|
|
pending->pending.owningArena = nullptr;
|
2022-06-17 01:52:23 +01:00
|
|
|
}
|
|
|
|
|
2022-01-06 22:10:07 +00:00
|
|
|
return pending.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
PendingType* TxnLog::pending(TypeId ty) const
|
|
|
|
{
|
2022-02-11 18:43:14 +00:00
|
|
|
// This function will technically work if `this` is nullptr, but this
|
|
|
|
// indicates a bug, so we explicitly assert.
|
|
|
|
LUAU_ASSERT(static_cast<const void*>(this) != nullptr);
|
|
|
|
|
2022-01-06 22:10:07 +00:00
|
|
|
for (const TxnLog* current = this; current; current = current->parent)
|
|
|
|
{
|
2023-05-12 13:15:01 +01:00
|
|
|
if (auto it = current->typeVarChanges.find(ty); it && !(*it)->dead)
|
2022-04-14 22:57:15 +01:00
|
|
|
return it->get();
|
2022-01-06 22:10:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
PendingTypePack* TxnLog::pending(TypePackId tp) const
|
|
|
|
{
|
2022-02-11 18:43:14 +00:00
|
|
|
// This function will technically work if `this` is nullptr, but this
|
|
|
|
// indicates a bug, so we explicitly assert.
|
|
|
|
LUAU_ASSERT(static_cast<const void*>(this) != nullptr);
|
|
|
|
|
2022-01-06 22:10:07 +00:00
|
|
|
for (const TxnLog* current = this; current; current = current->parent)
|
|
|
|
{
|
2022-04-14 22:57:15 +01:00
|
|
|
if (auto it = current->typePackChanges.find(tp))
|
|
|
|
return it->get();
|
2022-01-06 22:10:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2023-01-03 17:33:19 +00:00
|
|
|
PendingType* TxnLog::replace(TypeId ty, Type replacement)
|
2022-01-06 22:10:07 +00:00
|
|
|
{
|
|
|
|
PendingType* newTy = queue(ty);
|
2022-07-08 02:05:31 +01:00
|
|
|
newTy->pending.reassign(replacement);
|
2022-01-06 22:10:07 +00:00
|
|
|
return newTy;
|
|
|
|
}
|
|
|
|
|
|
|
|
PendingTypePack* TxnLog::replace(TypePackId tp, TypePackVar replacement)
|
|
|
|
{
|
|
|
|
PendingTypePack* newTp = queue(tp);
|
2022-07-08 02:05:31 +01:00
|
|
|
newTp->pending.reassign(replacement);
|
2022-01-06 22:10:07 +00:00
|
|
|
return newTp;
|
|
|
|
}
|
|
|
|
|
|
|
|
PendingType* TxnLog::bindTable(TypeId ty, std::optional<TypeId> newBoundTo)
|
|
|
|
{
|
2023-01-03 17:33:19 +00:00
|
|
|
LUAU_ASSERT(get<TableType>(ty));
|
2023-03-10 19:20:04 +00:00
|
|
|
LUAU_ASSERT(ty != newBoundTo);
|
2022-01-06 22:10:07 +00:00
|
|
|
|
|
|
|
PendingType* newTy = queue(ty);
|
2023-01-03 17:33:19 +00:00
|
|
|
if (TableType* ttv = Luau::getMutable<TableType>(newTy))
|
2022-01-06 22:10:07 +00:00
|
|
|
ttv->boundTo = newBoundTo;
|
|
|
|
|
|
|
|
return newTy;
|
|
|
|
}
|
|
|
|
|
|
|
|
PendingType* TxnLog::changeLevel(TypeId ty, TypeLevel newLevel)
|
|
|
|
{
|
2023-01-03 17:33:19 +00:00
|
|
|
LUAU_ASSERT(get<FreeType>(ty) || get<TableType>(ty) || get<FunctionType>(ty));
|
2022-01-06 22:10:07 +00:00
|
|
|
|
|
|
|
PendingType* newTy = queue(ty);
|
2023-01-03 17:33:19 +00:00
|
|
|
if (FreeType* ftv = Luau::getMutable<FreeType>(newTy))
|
2022-01-06 22:10:07 +00:00
|
|
|
{
|
|
|
|
ftv->level = newLevel;
|
|
|
|
}
|
2023-01-03 17:33:19 +00:00
|
|
|
else if (TableType* ttv = Luau::getMutable<TableType>(newTy))
|
2022-01-06 22:10:07 +00:00
|
|
|
{
|
|
|
|
LUAU_ASSERT(ttv->state == TableState::Free || ttv->state == TableState::Generic);
|
|
|
|
ttv->level = newLevel;
|
|
|
|
}
|
2023-01-03 17:33:19 +00:00
|
|
|
else if (FunctionType* ftv = Luau::getMutable<FunctionType>(newTy))
|
2022-01-06 22:10:07 +00:00
|
|
|
{
|
|
|
|
ftv->level = newLevel;
|
|
|
|
}
|
|
|
|
|
|
|
|
return newTy;
|
|
|
|
}
|
|
|
|
|
|
|
|
PendingTypePack* TxnLog::changeLevel(TypePackId tp, TypeLevel newLevel)
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(get<FreeTypePack>(tp));
|
|
|
|
|
|
|
|
PendingTypePack* newTp = queue(tp);
|
|
|
|
if (FreeTypePack* ftp = Luau::getMutable<FreeTypePack>(newTp))
|
|
|
|
{
|
|
|
|
ftp->level = newLevel;
|
|
|
|
}
|
|
|
|
|
|
|
|
return newTp;
|
|
|
|
}
|
|
|
|
|
2022-09-29 23:11:54 +01:00
|
|
|
PendingType* TxnLog::changeScope(TypeId ty, NotNull<Scope> newScope)
|
|
|
|
{
|
2023-01-03 17:33:19 +00:00
|
|
|
LUAU_ASSERT(get<FreeType>(ty) || get<TableType>(ty) || get<FunctionType>(ty));
|
2022-09-29 23:11:54 +01:00
|
|
|
|
|
|
|
PendingType* newTy = queue(ty);
|
2023-01-03 17:33:19 +00:00
|
|
|
if (FreeType* ftv = Luau::getMutable<FreeType>(newTy))
|
2022-09-29 23:11:54 +01:00
|
|
|
{
|
|
|
|
ftv->scope = newScope;
|
|
|
|
}
|
2023-01-03 17:33:19 +00:00
|
|
|
else if (TableType* ttv = Luau::getMutable<TableType>(newTy))
|
2022-09-29 23:11:54 +01:00
|
|
|
{
|
|
|
|
LUAU_ASSERT(ttv->state == TableState::Free || ttv->state == TableState::Generic);
|
|
|
|
ttv->scope = newScope;
|
|
|
|
}
|
2023-01-03 17:33:19 +00:00
|
|
|
else if (FunctionType* ftv = Luau::getMutable<FunctionType>(newTy))
|
2022-09-29 23:11:54 +01:00
|
|
|
{
|
|
|
|
ftv->scope = newScope;
|
|
|
|
}
|
|
|
|
|
|
|
|
return newTy;
|
|
|
|
}
|
|
|
|
|
|
|
|
PendingTypePack* TxnLog::changeScope(TypePackId tp, NotNull<Scope> newScope)
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(get<FreeTypePack>(tp));
|
|
|
|
|
|
|
|
PendingTypePack* newTp = queue(tp);
|
|
|
|
if (FreeTypePack* ftp = Luau::getMutable<FreeTypePack>(newTp))
|
|
|
|
{
|
|
|
|
ftp->scope = newScope;
|
|
|
|
}
|
|
|
|
|
|
|
|
return newTp;
|
|
|
|
}
|
|
|
|
|
2022-01-06 22:10:07 +00:00
|
|
|
PendingType* TxnLog::changeIndexer(TypeId ty, std::optional<TableIndexer> indexer)
|
|
|
|
{
|
2023-01-03 17:33:19 +00:00
|
|
|
LUAU_ASSERT(get<TableType>(ty));
|
2022-01-06 22:10:07 +00:00
|
|
|
|
|
|
|
PendingType* newTy = queue(ty);
|
2023-01-03 17:33:19 +00:00
|
|
|
if (TableType* ttv = Luau::getMutable<TableType>(newTy))
|
2022-01-06 22:10:07 +00:00
|
|
|
{
|
|
|
|
ttv->indexer = indexer;
|
|
|
|
}
|
|
|
|
|
|
|
|
return newTy;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<TypeLevel> TxnLog::getLevel(TypeId ty) const
|
|
|
|
{
|
2023-01-03 17:33:19 +00:00
|
|
|
if (FreeType* ftv = getMutable<FreeType>(ty))
|
2022-01-06 22:10:07 +00:00
|
|
|
return ftv->level;
|
2023-01-03 17:33:19 +00:00
|
|
|
else if (TableType* ttv = getMutable<TableType>(ty); ttv && (ttv->state == TableState::Free || ttv->state == TableState::Generic))
|
2022-01-06 22:10:07 +00:00
|
|
|
return ttv->level;
|
2023-01-03 17:33:19 +00:00
|
|
|
else if (FunctionType* ftv = getMutable<FunctionType>(ty))
|
2022-01-06 22:10:07 +00:00
|
|
|
return ftv->level;
|
|
|
|
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
2022-02-03 23:09:37 +00:00
|
|
|
TypeId TxnLog::follow(TypeId ty) const
|
2022-01-06 22:10:07 +00:00
|
|
|
{
|
2023-05-05 20:57:12 +01:00
|
|
|
return Luau::follow(ty, this, [](const void* ctx, TypeId ty) -> TypeId {
|
|
|
|
const TxnLog* self = static_cast<const TxnLog*>(ctx);
|
|
|
|
PendingType* state = self->pending(ty);
|
2022-01-06 22:10:07 +00:00
|
|
|
|
|
|
|
if (state == nullptr)
|
|
|
|
return ty;
|
|
|
|
|
|
|
|
// Ugly: Fabricate a TypeId that doesn't adhere to most of the invariants
|
|
|
|
// that normally apply. This is safe because follow will only call get<>
|
|
|
|
// on the returned pointer.
|
2023-01-03 17:33:19 +00:00
|
|
|
return const_cast<const Type*>(&state->pending);
|
2022-01-06 22:10:07 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
TypePackId TxnLog::follow(TypePackId tp) const
|
|
|
|
{
|
2023-05-05 20:57:12 +01:00
|
|
|
return Luau::follow(tp, this, [](const void* ctx, TypePackId tp) -> TypePackId {
|
|
|
|
const TxnLog* self = static_cast<const TxnLog*>(ctx);
|
|
|
|
PendingTypePack* state = self->pending(tp);
|
2022-01-06 22:10:07 +00:00
|
|
|
|
|
|
|
if (state == nullptr)
|
|
|
|
return tp;
|
|
|
|
|
|
|
|
// Ugly: Fabricate a TypePackId that doesn't adhere to most of the
|
|
|
|
// invariants that normally apply. This is safe because follow will
|
|
|
|
// only call get<> on the returned pointer.
|
|
|
|
return const_cast<const TypePackVar*>(&state->pending);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-08-25 21:55:08 +01:00
|
|
|
std::pair<std::vector<TypeId>, std::vector<TypePackId>> TxnLog::getChanges() const
|
|
|
|
{
|
|
|
|
std::pair<std::vector<TypeId>, std::vector<TypePackId>> result;
|
|
|
|
|
|
|
|
for (const auto& [typeId, _newState] : typeVarChanges)
|
|
|
|
result.first.push_back(typeId);
|
|
|
|
for (const auto& [typePackId, _newState] : typePackChanges)
|
|
|
|
result.second.push_back(typePackId);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
} // namespace Luau
|