// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details #include "Luau/Substitution.h" #include "Luau/Common.h" #include "Luau/Clone.h" #include "Luau/TxnLog.h" #include #include LUAU_FASTFLAG(LuauClonePublicInterfaceLess2) LUAU_FASTINTVARIABLE(LuauTarjanChildLimit, 10000) LUAU_FASTFLAG(DebugLuauReadWriteProperties) LUAU_FASTFLAG(LuauCloneSkipNonInternalVisit) LUAU_FASTFLAGVARIABLE(LuauTarjanSingleArr, false) namespace Luau { static TypeId DEPRECATED_shallowClone(TypeId ty, TypeArena& dest, const TxnLog* log, bool alwaysClone) { ty = log->follow(ty); TypeId result = ty; if (auto pty = log->pending(ty)) ty = &pty->pending; if (const FunctionType* ftv = get(ty)) { FunctionType clone = FunctionType{ftv->level, ftv->scope, ftv->argTypes, ftv->retTypes, ftv->definition, ftv->hasSelf}; clone.generics = ftv->generics; clone.genericPacks = ftv->genericPacks; clone.magicFunction = ftv->magicFunction; clone.dcrMagicFunction = ftv->dcrMagicFunction; clone.dcrMagicRefinement = ftv->dcrMagicRefinement; clone.tags = ftv->tags; clone.argNames = ftv->argNames; result = dest.addType(std::move(clone)); } else if (const TableType* ttv = get(ty)) { LUAU_ASSERT(!ttv->boundTo); TableType clone = TableType{ttv->props, ttv->indexer, ttv->level, ttv->scope, ttv->state}; clone.definitionModuleName = ttv->definitionModuleName; clone.definitionLocation = ttv->definitionLocation; clone.name = ttv->name; clone.syntheticName = ttv->syntheticName; clone.instantiatedTypeParams = ttv->instantiatedTypeParams; clone.instantiatedTypePackParams = ttv->instantiatedTypePackParams; clone.tags = ttv->tags; result = dest.addType(std::move(clone)); } else if (const MetatableType* mtv = get(ty)) { MetatableType clone = MetatableType{mtv->table, mtv->metatable}; clone.syntheticName = mtv->syntheticName; result = dest.addType(std::move(clone)); } else if (const UnionType* utv = get(ty)) { UnionType clone; clone.options = utv->options; result = dest.addType(std::move(clone)); } else if (const IntersectionType* itv = get(ty)) { IntersectionType clone; clone.parts = itv->parts; result = dest.addType(std::move(clone)); } else if (const PendingExpansionType* petv = get(ty)) { PendingExpansionType clone{petv->prefix, petv->name, petv->typeArguments, petv->packArguments}; result = dest.addType(std::move(clone)); } else if (const NegationType* ntv = get(ty)) { result = dest.addType(NegationType{ntv->ty}); } else if (const TypeFamilyInstanceType* tfit = get(ty)) { TypeFamilyInstanceType clone{tfit->family, tfit->typeArguments, tfit->packArguments}; result = dest.addType(std::move(clone)); } else return result; asMutable(result)->documentationSymbol = ty->documentationSymbol; return result; } static TypeId shallowClone(TypeId ty, TypeArena& dest, const TxnLog* log, bool alwaysClone) { if (!FFlag::LuauClonePublicInterfaceLess2) return DEPRECATED_shallowClone(ty, dest, log, alwaysClone); auto go = [ty, &dest, alwaysClone](auto&& a) { using T = std::decay_t; if constexpr (std::is_same_v) return ty; else if constexpr (std::is_same_v) { // This should never happen, but visit() cannot see it. LUAU_ASSERT(!"shallowClone didn't follow its argument!"); return dest.addType(BoundType{a.boundTo}); } else if constexpr (std::is_same_v) return dest.addType(a); else if constexpr (std::is_same_v) return dest.addType(a); else if constexpr (std::is_same_v) { LUAU_ASSERT(ty->persistent); return ty; } else if constexpr (std::is_same_v) { PendingExpansionType clone = PendingExpansionType{a.prefix, a.name, a.typeArguments, a.packArguments}; return dest.addType(std::move(clone)); } else if constexpr (std::is_same_v) { LUAU_ASSERT(ty->persistent); return ty; } else if constexpr (std::is_same_v) { LUAU_ASSERT(ty->persistent); return ty; } else if constexpr (std::is_same_v) { LUAU_ASSERT(ty->persistent); return ty; } else if constexpr (std::is_same_v) { LUAU_ASSERT(ty->persistent); return ty; } else if constexpr (std::is_same_v) return ty; else if constexpr (std::is_same_v) return dest.addType(a); else if constexpr (std::is_same_v) { FunctionType clone = FunctionType{a.level, a.scope, a.argTypes, a.retTypes, a.definition, a.hasSelf}; clone.generics = a.generics; clone.genericPacks = a.genericPacks; clone.magicFunction = a.magicFunction; clone.dcrMagicFunction = a.dcrMagicFunction; clone.dcrMagicRefinement = a.dcrMagicRefinement; clone.tags = a.tags; clone.argNames = a.argNames; return dest.addType(std::move(clone)); } else if constexpr (std::is_same_v) { LUAU_ASSERT(!a.boundTo); TableType clone = TableType{a.props, a.indexer, a.level, a.scope, a.state}; clone.definitionModuleName = a.definitionModuleName; clone.definitionLocation = a.definitionLocation; clone.name = a.name; clone.syntheticName = a.syntheticName; clone.instantiatedTypeParams = a.instantiatedTypeParams; clone.instantiatedTypePackParams = a.instantiatedTypePackParams; clone.tags = a.tags; return dest.addType(std::move(clone)); } else if constexpr (std::is_same_v) { MetatableType clone = MetatableType{a.table, a.metatable}; clone.syntheticName = a.syntheticName; return dest.addType(std::move(clone)); } else if constexpr (std::is_same_v) { UnionType clone; clone.options = a.options; return dest.addType(std::move(clone)); } else if constexpr (std::is_same_v) { IntersectionType clone; clone.parts = a.parts; return dest.addType(std::move(clone)); } else if constexpr (std::is_same_v) { if (alwaysClone) { ClassType clone{a.name, a.props, a.parent, a.metatable, a.tags, a.userData, a.definitionModuleName, a.indexer}; return dest.addType(std::move(clone)); } else return ty; } else if constexpr (std::is_same_v) return dest.addType(NegationType{a.ty}); else if constexpr (std::is_same_v) { TypeFamilyInstanceType clone{a.family, a.typeArguments, a.packArguments}; return dest.addType(std::move(clone)); } else static_assert(always_false_v, "Non-exhaustive shallowClone switch"); }; ty = log->follow(ty); if (auto pty = log->pending(ty)) ty = &pty->pending; TypeId resTy = visit(go, ty->ty); if (resTy != ty) asMutable(resTy)->documentationSymbol = ty->documentationSymbol; return resTy; } void Tarjan::visitChildren(TypeId ty, int index) { LUAU_ASSERT(ty == log->follow(ty)); if (FFlag::LuauCloneSkipNonInternalVisit ? ignoreChildrenVisit(ty) : ignoreChildren(ty)) return; if (auto pty = log->pending(ty)) ty = &pty->pending; if (const FunctionType* ftv = get(ty)) { for (TypeId generic : ftv->generics) visitChild(generic); for (TypePackId genericPack : ftv->genericPacks) visitChild(genericPack); visitChild(ftv->argTypes); visitChild(ftv->retTypes); } else if (const TableType* ttv = get(ty)) { LUAU_ASSERT(!ttv->boundTo); for (const auto& [name, prop] : ttv->props) { if (FFlag::DebugLuauReadWriteProperties) { visitChild(prop.readType()); visitChild(prop.writeType()); } else visitChild(prop.type()); } if (ttv->indexer) { visitChild(ttv->indexer->indexType); visitChild(ttv->indexer->indexResultType); } for (TypeId itp : ttv->instantiatedTypeParams) visitChild(itp); for (TypePackId itp : ttv->instantiatedTypePackParams) visitChild(itp); } else if (const MetatableType* mtv = get(ty)) { visitChild(mtv->table); visitChild(mtv->metatable); } else if (const UnionType* utv = get(ty)) { for (TypeId opt : utv->options) visitChild(opt); } else if (const IntersectionType* itv = get(ty)) { for (TypeId part : itv->parts) visitChild(part); } else if (const PendingExpansionType* petv = get(ty)) { for (TypeId a : petv->typeArguments) visitChild(a); for (TypePackId a : petv->packArguments) visitChild(a); } else if (const TypeFamilyInstanceType* tfit = get(ty)) { for (TypeId a : tfit->typeArguments) visitChild(a); for (TypePackId a : tfit->packArguments) visitChild(a); } else if (const ClassType* ctv = get(ty)) { for (const auto& [name, prop] : ctv->props) visitChild(prop.type()); if (ctv->parent) visitChild(*ctv->parent); if (ctv->metatable) visitChild(*ctv->metatable); if (ctv->indexer) { visitChild(ctv->indexer->indexType); visitChild(ctv->indexer->indexResultType); } } else if (const NegationType* ntv = get(ty)) { visitChild(ntv->ty); } } void Tarjan::visitChildren(TypePackId tp, int index) { LUAU_ASSERT(tp == log->follow(tp)); if (FFlag::LuauCloneSkipNonInternalVisit ? ignoreChildrenVisit(tp) : ignoreChildren(tp)) return; if (auto ptp = log->pending(tp)) tp = &ptp->pending; if (const TypePack* tpp = get(tp)) { for (TypeId tv : tpp->head) visitChild(tv); if (tpp->tail) visitChild(*tpp->tail); } else if (const VariadicTypePack* vtp = get(tp)) { visitChild(vtp->ty); } } std::pair Tarjan::indexify(TypeId ty) { ty = log->follow(ty); if (FFlag::LuauTarjanSingleArr) { auto [index, fresh] = typeToIndex.try_insert(ty, false); if (fresh) { index = int(nodes.size()); nodes.push_back({ty, nullptr, false, false, index}); } return {index, fresh}; } else { bool fresh = !typeToIndex.contains(ty); int& index = typeToIndex[ty]; if (fresh) { index = int(indexToType.size()); indexToType.push_back(ty); indexToPack.push_back(nullptr); onStack.push_back(false); lowlink.push_back(index); } return {index, fresh}; } } std::pair Tarjan::indexify(TypePackId tp) { tp = log->follow(tp); if (FFlag::LuauTarjanSingleArr) { auto [index, fresh] = packToIndex.try_insert(tp, false); if (fresh) { index = int(nodes.size()); nodes.push_back({nullptr, tp, false, false, index}); } return {index, fresh}; } else { bool fresh = !packToIndex.contains(tp); int& index = packToIndex[tp]; if (fresh) { index = int(indexToPack.size()); indexToType.push_back(nullptr); indexToPack.push_back(tp); onStack.push_back(false); lowlink.push_back(index); } return {index, fresh}; } } void Tarjan::visitChild(TypeId ty) { ty = log->follow(ty); edgesTy.push_back(ty); edgesTp.push_back(nullptr); } void Tarjan::visitChild(TypePackId tp) { tp = log->follow(tp); edgesTy.push_back(nullptr); edgesTp.push_back(tp); } TarjanResult Tarjan::loop() { if (!FFlag::LuauTarjanSingleArr) return loop_DEPRECATED(); // Normally Tarjan is presented recursively, but this is a hot loop, so worth optimizing while (!worklist.empty()) { auto [index, currEdge, lastEdge] = worklist.back(); // First visit if (currEdge == -1) { ++childCount; if (childLimit > 0 && childLimit <= childCount) return TarjanResult::TooManyChildren; stack.push_back(index); nodes[index].onStack = true; currEdge = int(edgesTy.size()); // Fill in edge list of this vertex if (TypeId ty = nodes[index].ty) visitChildren(ty, index); else if (TypePackId tp = nodes[index].tp) visitChildren(tp, index); lastEdge = int(edgesTy.size()); } // Visit children bool foundFresh = false; for (; currEdge < lastEdge; currEdge++) { int childIndex = -1; bool fresh = false; if (auto ty = edgesTy[currEdge]) std::tie(childIndex, fresh) = indexify(ty); else if (auto tp = edgesTp[currEdge]) std::tie(childIndex, fresh) = indexify(tp); else LUAU_ASSERT(false); if (fresh) { // Original recursion point, update the parent continuation point and start the new element worklist.back() = {index, currEdge + 1, lastEdge}; worklist.push_back({childIndex, -1, -1}); // We need to continue the top-level loop from the start with the new worklist element foundFresh = true; break; } else if (nodes[childIndex].onStack) { nodes[index].lowlink = std::min(nodes[index].lowlink, childIndex); } visitEdge(childIndex, index); } if (foundFresh) continue; if (nodes[index].lowlink == index) { visitSCC(index); while (!stack.empty()) { int popped = stack.back(); stack.pop_back(); nodes[popped].onStack = false; if (popped == index) break; } } worklist.pop_back(); // Original return from recursion into a child if (!worklist.empty()) { auto [parentIndex, _, parentEndEdge] = worklist.back(); // No need to keep child edges around edgesTy.resize(parentEndEdge); edgesTp.resize(parentEndEdge); nodes[parentIndex].lowlink = std::min(nodes[parentIndex].lowlink, nodes[index].lowlink); visitEdge(index, parentIndex); } } return TarjanResult::Ok; } TarjanResult Tarjan::visitRoot(TypeId ty) { childCount = 0; if (childLimit == 0) childLimit = FInt::LuauTarjanChildLimit; ty = log->follow(ty); auto [index, fresh] = indexify(ty); worklist.push_back({index, -1, -1}); return loop(); } TarjanResult Tarjan::visitRoot(TypePackId tp) { childCount = 0; if (childLimit == 0) childLimit = FInt::LuauTarjanChildLimit; tp = log->follow(tp); auto [index, fresh] = indexify(tp); worklist.push_back({index, -1, -1}); return loop(); } void Tarjan::clearTarjan() { if (FFlag::LuauTarjanSingleArr) { typeToIndex.clear(); packToIndex.clear(); nodes.clear(); stack.clear(); } else { dirty.clear(); typeToIndex.clear(); packToIndex.clear(); indexToType.clear(); indexToPack.clear(); stack.clear(); onStack.clear(); lowlink.clear(); } edgesTy.clear(); edgesTp.clear(); worklist.clear(); } bool Tarjan::getDirty(int index) { if (FFlag::LuauTarjanSingleArr) { LUAU_ASSERT(size_t(index) < nodes.size()); return nodes[index].dirty; } else { if (dirty.size() <= size_t(index)) dirty.resize(index + 1, false); return dirty[index]; } } void Tarjan::setDirty(int index, bool d) { if (FFlag::LuauTarjanSingleArr) { LUAU_ASSERT(size_t(index) < nodes.size()); nodes[index].dirty = d; } else { if (dirty.size() <= size_t(index)) dirty.resize(index + 1, false); dirty[index] = d; } } void Tarjan::visitEdge(int index, int parentIndex) { if (getDirty(index)) setDirty(parentIndex, true); } void Tarjan::visitSCC(int index) { if (!FFlag::LuauTarjanSingleArr) return visitSCC_DEPRECATED(index); bool d = getDirty(index); for (auto it = stack.rbegin(); !d && it != stack.rend(); it++) { TarjanNode& node = nodes[*it]; if (TypeId ty = node.ty) d = isDirty(ty); else if (TypePackId tp = node.tp) d = isDirty(tp); if (*it == index) break; } if (!d) return; for (auto it = stack.rbegin(); it != stack.rend(); it++) { setDirty(*it, true); TarjanNode& node = nodes[*it]; if (TypeId ty = node.ty) foundDirty(ty); else if (TypePackId tp = node.tp) foundDirty(tp); if (*it == index) return; } } TarjanResult Tarjan::findDirty(TypeId ty) { return visitRoot(ty); } TarjanResult Tarjan::findDirty(TypePackId tp) { return visitRoot(tp); } TarjanResult Tarjan::loop_DEPRECATED() { // Normally Tarjan is presented recursively, but this is a hot loop, so worth optimizing while (!worklist.empty()) { auto [index, currEdge, lastEdge] = worklist.back(); // First visit if (currEdge == -1) { ++childCount; if (childLimit > 0 && childLimit <= childCount) return TarjanResult::TooManyChildren; stack.push_back(index); onStack[index] = true; currEdge = int(edgesTy.size()); // Fill in edge list of this vertex if (TypeId ty = indexToType[index]) visitChildren(ty, index); else if (TypePackId tp = indexToPack[index]) visitChildren(tp, index); lastEdge = int(edgesTy.size()); } // Visit children bool foundFresh = false; for (; currEdge < lastEdge; currEdge++) { int childIndex = -1; bool fresh = false; if (auto ty = edgesTy[currEdge]) std::tie(childIndex, fresh) = indexify(ty); else if (auto tp = edgesTp[currEdge]) std::tie(childIndex, fresh) = indexify(tp); else LUAU_ASSERT(false); if (fresh) { // Original recursion point, update the parent continuation point and start the new element worklist.back() = {index, currEdge + 1, lastEdge}; worklist.push_back({childIndex, -1, -1}); // We need to continue the top-level loop from the start with the new worklist element foundFresh = true; break; } else if (onStack[childIndex]) { lowlink[index] = std::min(lowlink[index], childIndex); } visitEdge(childIndex, index); } if (foundFresh) continue; if (lowlink[index] == index) { visitSCC(index); while (!stack.empty()) { int popped = stack.back(); stack.pop_back(); onStack[popped] = false; if (popped == index) break; } } worklist.pop_back(); // Original return from recursion into a child if (!worklist.empty()) { auto [parentIndex, _, parentEndEdge] = worklist.back(); // No need to keep child edges around edgesTy.resize(parentEndEdge); edgesTp.resize(parentEndEdge); lowlink[parentIndex] = std::min(lowlink[parentIndex], lowlink[index]); visitEdge(index, parentIndex); } } return TarjanResult::Ok; } void Tarjan::visitSCC_DEPRECATED(int index) { bool d = getDirty(index); for (auto it = stack.rbegin(); !d && it != stack.rend(); it++) { if (TypeId ty = indexToType[*it]) d = isDirty(ty); else if (TypePackId tp = indexToPack[*it]) d = isDirty(tp); if (*it == index) break; } if (!d) return; for (auto it = stack.rbegin(); it != stack.rend(); it++) { setDirty(*it, true); if (TypeId ty = indexToType[*it]) foundDirty(ty); else if (TypePackId tp = indexToPack[*it]) foundDirty(tp); if (*it == index) return; } } std::optional Substitution::substitute(TypeId ty) { ty = log->follow(ty); // clear algorithm state for reentrancy clearTarjan(); auto result = findDirty(ty); if (result != TarjanResult::Ok) return std::nullopt; for (auto [oldTy, newTy] : newTypes) { if (!ignoreChildren(oldTy) && !replacedTypes.contains(newTy)) { replaceChildren(newTy); replacedTypes.insert(newTy); } } for (auto [oldTp, newTp] : newPacks) { if (!ignoreChildren(oldTp) && !replacedTypePacks.contains(newTp)) { replaceChildren(newTp); replacedTypePacks.insert(newTp); } } TypeId newTy = replace(ty); return newTy; } std::optional Substitution::substitute(TypePackId tp) { tp = log->follow(tp); // clear algorithm state for reentrancy clearTarjan(); auto result = findDirty(tp); if (result != TarjanResult::Ok) return std::nullopt; for (auto [oldTy, newTy] : newTypes) { if (!ignoreChildren(oldTy) && !replacedTypes.contains(newTy)) { replaceChildren(newTy); replacedTypes.insert(newTy); } } for (auto [oldTp, newTp] : newPacks) { if (!ignoreChildren(oldTp) && !replacedTypePacks.contains(newTp)) { replaceChildren(newTp); replacedTypePacks.insert(newTp); } } TypePackId newTp = replace(tp); return newTp; } TypeId Substitution::clone(TypeId ty) { return shallowClone(ty, *arena, log, /* alwaysClone */ FFlag::LuauClonePublicInterfaceLess2); } TypePackId Substitution::clone(TypePackId tp) { tp = log->follow(tp); if (auto ptp = log->pending(tp)) tp = &ptp->pending; if (const TypePack* tpp = get(tp)) { TypePack clone; clone.head = tpp->head; clone.tail = tpp->tail; return addTypePack(std::move(clone)); } else if (const VariadicTypePack* vtp = get(tp)) { VariadicTypePack clone; clone.ty = vtp->ty; clone.hidden = vtp->hidden; return addTypePack(std::move(clone)); } else if (const TypeFamilyInstanceTypePack* tfitp = get(tp)) { TypeFamilyInstanceTypePack clone{ tfitp->family, std::vector(tfitp->typeArguments.size()), std::vector(tfitp->packArguments.size())}; clone.typeArguments.assign(tfitp->typeArguments.begin(), tfitp->typeArguments.end()); clone.packArguments.assign(tfitp->packArguments.begin(), tfitp->packArguments.end()); return addTypePack(std::move(clone)); } else if (FFlag::LuauClonePublicInterfaceLess2) { return addTypePack(*tp); } else return tp; } void Substitution::foundDirty(TypeId ty) { ty = log->follow(ty); if (newTypes.contains(ty)) return; if (isDirty(ty)) newTypes[ty] = follow(clean(ty)); else newTypes[ty] = follow(clone(ty)); } void Substitution::foundDirty(TypePackId tp) { tp = log->follow(tp); if (newPacks.contains(tp)) return; if (isDirty(tp)) newPacks[tp] = follow(clean(tp)); else newPacks[tp] = follow(clone(tp)); } TypeId Substitution::replace(TypeId ty) { ty = log->follow(ty); if (TypeId* prevTy = newTypes.find(ty)) return *prevTy; else return ty; } TypePackId Substitution::replace(TypePackId tp) { tp = log->follow(tp); if (TypePackId* prevTp = newPacks.find(tp)) return *prevTp; else return tp; } void Substitution::replaceChildren(TypeId ty) { LUAU_ASSERT(ty == log->follow(ty)); if (ignoreChildren(ty)) return; if (ty->owningArena != arena) return; if (FunctionType* ftv = getMutable(ty)) { for (TypeId& generic : ftv->generics) generic = replace(generic); for (TypePackId& genericPack : ftv->genericPacks) genericPack = replace(genericPack); ftv->argTypes = replace(ftv->argTypes); ftv->retTypes = replace(ftv->retTypes); } else if (TableType* ttv = getMutable(ty)) { LUAU_ASSERT(!ttv->boundTo); for (auto& [name, prop] : ttv->props) { if (FFlag::DebugLuauReadWriteProperties) prop = Property::create(replace(prop.readType()), replace(prop.writeType())); else prop.setType(replace(prop.type())); } if (ttv->indexer) { ttv->indexer->indexType = replace(ttv->indexer->indexType); ttv->indexer->indexResultType = replace(ttv->indexer->indexResultType); } for (TypeId& itp : ttv->instantiatedTypeParams) itp = replace(itp); for (TypePackId& itp : ttv->instantiatedTypePackParams) itp = replace(itp); } else if (MetatableType* mtv = getMutable(ty)) { mtv->table = replace(mtv->table); mtv->metatable = replace(mtv->metatable); } else if (UnionType* utv = getMutable(ty)) { for (TypeId& opt : utv->options) opt = replace(opt); } else if (IntersectionType* itv = getMutable(ty)) { for (TypeId& part : itv->parts) part = replace(part); } else if (PendingExpansionType* petv = getMutable(ty)) { for (TypeId& a : petv->typeArguments) a = replace(a); for (TypePackId& a : petv->packArguments) a = replace(a); } else if (TypeFamilyInstanceType* tfit = getMutable(ty)) { for (TypeId& a : tfit->typeArguments) a = replace(a); for (TypePackId& a : tfit->packArguments) a = replace(a); } else if (ClassType* ctv = getMutable(ty)) { for (auto& [name, prop] : ctv->props) prop.setType(replace(prop.type())); if (ctv->parent) ctv->parent = replace(*ctv->parent); if (ctv->metatable) ctv->metatable = replace(*ctv->metatable); if (ctv->indexer) { ctv->indexer->indexType = replace(ctv->indexer->indexType); ctv->indexer->indexResultType = replace(ctv->indexer->indexResultType); } } else if (NegationType* ntv = getMutable(ty)) { ntv->ty = replace(ntv->ty); } } void Substitution::replaceChildren(TypePackId tp) { LUAU_ASSERT(tp == log->follow(tp)); if (ignoreChildren(tp)) return; if (tp->owningArena != arena) return; if (TypePack* tpp = getMutable(tp)) { for (TypeId& tv : tpp->head) tv = replace(tv); if (tpp->tail) tpp->tail = replace(*tpp->tail); } else if (VariadicTypePack* vtp = getMutable(tp)) { vtp->ty = replace(vtp->ty); } else if (TypeFamilyInstanceTypePack* tfitp = getMutable(tp)) { for (TypeId& t : tfitp->typeArguments) t = replace(t); for (TypePackId& t : tfitp->packArguments) t = replace(t); } } } // namespace Luau