// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details #include "Luau/TypeReduction.h" #include "Luau/Common.h" #include "Luau/Error.h" #include "Luau/RecursionCounter.h" #include "Luau/VisitType.h" #include #include LUAU_FASTINTVARIABLE(LuauTypeReductionCartesianProductLimit, 100'000) LUAU_FASTINTVARIABLE(LuauTypeReductionRecursionLimit, 300) LUAU_FASTFLAGVARIABLE(DebugLuauDontReduceTypes, false) namespace Luau { namespace detail { bool TypeReductionMemoization::isIrreducible(TypeId ty) { ty = follow(ty); // Only does shallow check, the TypeReducer itself already does deep traversal. if (auto edge = types.find(ty); edge && edge->irreducible) return true; else if (get(ty) || get(ty) || get(ty)) return false; else if (auto tt = get(ty); tt && (tt->state == TableState::Free || tt->state == TableState::Unsealed)) return false; else return true; } bool TypeReductionMemoization::isIrreducible(TypePackId tp) { tp = follow(tp); // Only does shallow check, the TypeReducer itself already does deep traversal. if (auto edge = typePacks.find(tp); edge && edge->irreducible) return true; else if (get(tp) || get(tp)) return false; else if (auto vtp = get(tp)) return isIrreducible(vtp->ty); else return true; } TypeId TypeReductionMemoization::memoize(TypeId ty, TypeId reducedTy) { ty = follow(ty); reducedTy = follow(reducedTy); // The irreducibility of this [`reducedTy`] depends on whether its contents are themselves irreducible. // We don't need to recurse much further than that, because we already record the irreducibility from // the bottom up. bool irreducible = isIrreducible(reducedTy); if (auto it = get(reducedTy)) { for (TypeId part : it) irreducible &= isIrreducible(part); } else if (auto ut = get(reducedTy)) { for (TypeId option : ut) irreducible &= isIrreducible(option); } else if (auto tt = get(reducedTy)) { for (auto& [k, p] : tt->props) irreducible &= isIrreducible(p.type); if (tt->indexer) { irreducible &= isIrreducible(tt->indexer->indexType); irreducible &= isIrreducible(tt->indexer->indexResultType); } for (auto ta : tt->instantiatedTypeParams) irreducible &= isIrreducible(ta); for (auto tpa : tt->instantiatedTypePackParams) irreducible &= isIrreducible(tpa); } else if (auto mt = get(reducedTy)) { irreducible &= isIrreducible(mt->table); irreducible &= isIrreducible(mt->metatable); } else if (auto ft = get(reducedTy)) { irreducible &= isIrreducible(ft->argTypes); irreducible &= isIrreducible(ft->retTypes); } else if (auto nt = get(reducedTy)) irreducible &= isIrreducible(nt->ty); types[ty] = {reducedTy, irreducible}; types[reducedTy] = {reducedTy, irreducible}; return reducedTy; } TypePackId TypeReductionMemoization::memoize(TypePackId tp, TypePackId reducedTp) { tp = follow(tp); reducedTp = follow(reducedTp); bool irreducible = isIrreducible(reducedTp); TypePackIterator it = begin(tp); while (it != end(tp)) { irreducible &= isIrreducible(*it); ++it; } if (it.tail()) irreducible &= isIrreducible(*it.tail()); typePacks[tp] = {reducedTp, irreducible}; typePacks[reducedTp] = {reducedTp, irreducible}; return reducedTp; } std::optional> TypeReductionMemoization::memoizedof(TypeId ty) const { auto fetchContext = [this](TypeId ty) -> std::optional> { if (auto edge = types.find(ty)) return *edge; else return std::nullopt; }; TypeId currentTy = ty; std::optional> lastEdge; while (auto edge = fetchContext(currentTy)) { lastEdge = edge; if (edge->irreducible) return edge; else if (edge->type == currentTy) return edge; else currentTy = edge->type; } return lastEdge; } std::optional> TypeReductionMemoization::memoizedof(TypePackId tp) const { auto fetchContext = [this](TypePackId tp) -> std::optional> { if (auto edge = typePacks.find(tp)) return *edge; else return std::nullopt; }; TypePackId currentTp = tp; std::optional> lastEdge; while (auto edge = fetchContext(currentTp)) { lastEdge = edge; if (edge->irreducible) return edge; else if (edge->type == currentTp) return edge; else currentTp = edge->type; } return lastEdge; } } // namespace detail namespace { template std::pair get2(const Thing& one, const Thing& two) { const A* a = get(one); const B* b = get(two); return a && b ? std::make_pair(a, b) : std::make_pair(nullptr, nullptr); } struct TypeReducer { NotNull arena; NotNull builtinTypes; NotNull handle; NotNull memoization; DenseHashSet* cyclics; int depth = 0; TypeId reduce(TypeId ty); TypePackId reduce(TypePackId tp); std::optional intersectionType(TypeId left, TypeId right); std::optional unionType(TypeId left, TypeId right); TypeId tableType(TypeId ty); TypeId functionType(TypeId ty); TypeId negationType(TypeId ty); using BinaryFold = std::optional (TypeReducer::*)(TypeId, TypeId); using UnaryFold = TypeId (TypeReducer::*)(TypeId); template LUAU_NOINLINE std::pair copy(TypeId ty, const T* t) { ty = follow(ty); if (auto edge = memoization->memoizedof(ty)) return {edge->type, getMutable(edge->type)}; // We specifically do not want to use [`detail::TypeReductionMemoization::memoize`] because that will // potentially consider these copiedTy to be reducible, but we need this to resolve cyclic references // without attempting to recursively reduce it, causing copies of copies of copies of... TypeId copiedTy = arena->addType(*t); memoization->types[ty] = {copiedTy, true}; memoization->types[copiedTy] = {copiedTy, true}; return {copiedTy, getMutable(copiedTy)}; } template void foldl_impl(Iter it, Iter endIt, BinaryFold f, std::vector* result, bool* didReduce) { RecursionLimiter rl{&depth, FInt::LuauTypeReductionRecursionLimit}; while (it != endIt) { TypeId right = reduce(*it); *didReduce |= right != follow(*it); // We're hitting a case where the `currentTy` returned a type that's the same as `T`. // e.g. `(string?) & ~(false | nil)` became `(string?) & (~false & ~nil)` but the current iterator we're consuming doesn't know this. // We will need to recurse and traverse that first. if (auto t = get(right)) { foldl_impl(begin(t), end(t), f, result, didReduce); ++it; continue; } bool replaced = false; auto resultIt = result->begin(); while (resultIt != result->end()) { TypeId left = *resultIt; if (left == right) { replaced = true; ++resultIt; continue; } std::optional reduced = (this->*f)(left, right); if (reduced) { *resultIt = *reduced; ++resultIt; replaced = true; } else { ++resultIt; continue; } } if (!replaced) result->push_back(right); *didReduce |= replaced; ++it; } } template TypeId flatten(std::vector&& types) { if (types.size() == 1) return types[0]; else return arena->addType(T{std::move(types)}); } template TypeId foldl(Iter it, Iter endIt, std::optional ty, BinaryFold f) { std::vector result; bool didReduce = false; foldl_impl(it, endIt, f, &result, &didReduce); // If we've done any reduction, then we'll need to reduce it again, e.g. // `"a" | "b" | string` is reduced into `string | string`, which is then reduced into `string`. if (!didReduce) return ty ? *ty : flatten(std::move(result)); else return reduce(flatten(std::move(result))); } template TypeId apply(BinaryFold f, TypeId left, TypeId right) { std::vector types{left, right}; return foldl(begin(types), end(types), std::nullopt, f); } template TypeId distribute(TypeIterator it, TypeIterator endIt, BinaryFold f, TypeId ty) { std::vector result; while (it != endIt) { result.push_back(apply(f, *it, ty)); ++it; } return flatten(std::move(result)); } }; TypeId TypeReducer::reduce(TypeId ty) { ty = follow(ty); if (auto edge = memoization->memoizedof(ty)) { if (edge->irreducible) return edge->type; else ty = follow(edge->type); } else if (cyclics->contains(ty)) return ty; RecursionLimiter rl{&depth, FInt::LuauTypeReductionRecursionLimit}; TypeId result = nullptr; if (auto i = get(ty)) result = foldl(begin(i), end(i), ty, &TypeReducer::intersectionType); else if (auto u = get(ty)) result = foldl(begin(u), end(u), ty, &TypeReducer::unionType); else if (get(ty) || get(ty)) result = tableType(ty); else if (get(ty)) result = functionType(ty); else if (get(ty)) result = negationType(ty); else result = ty; return memoization->memoize(ty, result); } TypePackId TypeReducer::reduce(TypePackId tp) { tp = follow(tp); if (auto edge = memoization->memoizedof(tp)) { if (edge->irreducible) return edge->type; else tp = edge->type; } else if (cyclics->contains(tp)) return tp; RecursionLimiter rl{&depth, FInt::LuauTypeReductionRecursionLimit}; bool didReduce = false; TypePackIterator it = begin(tp); std::vector head; while (it != end(tp)) { TypeId reducedTy = reduce(*it); head.push_back(reducedTy); didReduce |= follow(*it) != follow(reducedTy); ++it; } std::optional tail = it.tail(); if (tail) { if (auto vtp = get(follow(*it.tail()))) { TypeId reducedTy = reduce(vtp->ty); if (follow(vtp->ty) != follow(reducedTy)) { tail = arena->addTypePack(VariadicTypePack{reducedTy, vtp->hidden}); didReduce = true; } } } if (!didReduce) return memoization->memoize(tp, tp); else if (head.empty() && tail) return memoization->memoize(tp, *tail); else return memoization->memoize(tp, arena->addTypePack(TypePack{std::move(head), tail})); } std::optional TypeReducer::intersectionType(TypeId left, TypeId right) { if (get(left)) return left; // never & T ~ never else if (get(right)) return right; // T & never ~ never else if (get(left)) return right; // unknown & T ~ T else if (get(right)) return left; // T & unknown ~ T else if (get(left)) return right; // any & T ~ T else if (get(right)) return left; // T & any ~ T else if (get(left)) return std::nullopt; // 'a & T ~ 'a & T else if (get(right)) return std::nullopt; // T & 'a ~ T & 'a else if (get(left)) return std::nullopt; // G & T ~ G & T else if (get(right)) return std::nullopt; // T & G ~ T & G else if (get(left)) return std::nullopt; // error & T ~ error & T else if (get(right)) return std::nullopt; // T & error ~ T & error else if (get(left)) return std::nullopt; // *blocked* & T ~ *blocked* & T else if (get(right)) return std::nullopt; // T & *blocked* ~ T & *blocked* else if (get(left)) return std::nullopt; // *pending* & T ~ *pending* & T else if (get(right)) return std::nullopt; // T & *pending* ~ T & *pending* else if (auto [utl, utr] = get2(left, right); utl && utr) { std::vector parts; for (TypeId optionl : utl) { for (TypeId optionr : utr) parts.push_back(apply(&TypeReducer::intersectionType, optionl, optionr)); } return reduce(flatten(std::move(parts))); // (T | U) & (A | B) ~ (T & A) | (T & B) | (U & A) | (U & B) } else if (auto ut = get(left)) return reduce(distribute(begin(ut), end(ut), &TypeReducer::intersectionType, right)); // (A | B) & T ~ (A & T) | (B & T) else if (get(right)) return intersectionType(right, left); // T & (A | B) ~ (A | B) & T else if (auto [p1, p2] = get2(left, right); p1 && p2) { if (p1->type == p2->type) return left; // P1 & P2 ~ P1 iff P1 == P2 else return builtinTypes->neverType; // P1 & P2 ~ never iff P1 != P2 } else if (auto [p, s] = get2(left, right); p && s) { if (p->type == PrimitiveType::String && get(s)) return right; // string & "A" ~ "A" else if (p->type == PrimitiveType::Boolean && get(s)) return right; // boolean & true ~ true else return builtinTypes->neverType; // string & true ~ never } else if (auto [s, p] = get2(left, right); s && p) return intersectionType(right, left); // S & P ~ P & S else if (auto [p, f] = get2(left, right); p && f) { if (p->type == PrimitiveType::Function) return right; // function & () -> () ~ () -> () else return builtinTypes->neverType; // string & () -> () ~ never } else if (auto [f, p] = get2(left, right); f && p) return intersectionType(right, left); // () -> () & P ~ P & () -> () else if (auto [p, t] = get2(left, right); p && t) { if (p->type == PrimitiveType::Table) return right; // table & {} ~ {} else return builtinTypes->neverType; // string & {} ~ never } else if (auto [p, t] = get2(left, right); p && t) { if (p->type == PrimitiveType::Table) return right; // table & {} ~ {} else return builtinTypes->neverType; // string & {} ~ never } else if (auto [t, p] = get2(left, right); t && p) return intersectionType(right, left); // {} & P ~ P & {} else if (auto [t, p] = get2(left, right); t && p) return intersectionType(right, left); // M & P ~ P & M else if (auto [s1, s2] = get2(left, right); s1 && s2) { if (*s1 == *s2) return left; // "a" & "a" ~ "a" else return builtinTypes->neverType; // "a" & "b" ~ never } else if (auto [c1, c2] = get2(left, right); c1 && c2) { if (isSubclass(c1, c2)) return left; // Derived & Base ~ Derived else if (isSubclass(c2, c1)) return right; // Base & Derived ~ Derived else return builtinTypes->neverType; // Base & Unrelated ~ never } else if (auto [f1, f2] = get2(left, right); f1 && f2) return std::nullopt; // TODO else if (auto [t1, t2] = get2(left, right); t1 && t2) { if (t1->state == TableState::Free || t2->state == TableState::Free) return std::nullopt; // '{ x: T } & { x: U } ~ '{ x: T } & { x: U } else if (t1->state == TableState::Generic || t2->state == TableState::Generic) return std::nullopt; // '{ x: T } & { x: U } ~ '{ x: T } & { x: U } if (cyclics->contains(left)) return std::nullopt; // (t1 where t1 = { p: t1 }) & {} ~ t1 & {} else if (cyclics->contains(right)) return std::nullopt; // {} & (t1 where t1 = { p: t1 }) ~ {} & t1 TypeId resultTy = arena->addType(TableType{}); TableType* table = getMutable(resultTy); table->state = t1->state == TableState::Sealed || t2->state == TableState::Sealed ? TableState::Sealed : TableState::Unsealed; for (const auto& [name, prop] : t1->props) { // TODO: when t1 has properties, we should also intersect that with the indexer in t2 if it exists, // even if we have the corresponding property in the other one. if (auto other = t2->props.find(name); other != t2->props.end()) { TypeId propTy = apply(&TypeReducer::intersectionType, prop.type, other->second.type); if (get(propTy)) return builtinTypes->neverType; // { p : string } & { p : number } ~ { p : string & number } ~ { p : never } ~ never else table->props[name] = {propTy}; // { p : string } & { p : ~"a" } ~ { p : string & ~"a" } } else table->props[name] = prop; // { p : string } & {} ~ { p : string } } for (const auto& [name, prop] : t2->props) { // TODO: And vice versa, t2 properties against t1 indexer if it exists, // even if we have the corresponding property in the other one. if (!t1->props.count(name)) table->props[name] = {reduce(prop.type)}; // {} & { p : string & string } ~ { p : string } } if (t1->indexer && t2->indexer) { TypeId keyTy = apply(&TypeReducer::intersectionType, t1->indexer->indexType, t2->indexer->indexType); if (get(keyTy)) return std::nullopt; // { [string]: _ } & { [number]: _ } ~ { [string]: _ } & { [number]: _ } TypeId valueTy = apply(&TypeReducer::intersectionType, t1->indexer->indexResultType, t2->indexer->indexResultType); table->indexer = TableIndexer{keyTy, valueTy}; // { [string]: number } & { [string]: string } ~ { [string]: never } } else if (t1->indexer) { TypeId keyTy = reduce(t1->indexer->indexType); TypeId valueTy = reduce(t1->indexer->indexResultType); table->indexer = TableIndexer{keyTy, valueTy}; // { [number]: boolean } & { p : string } ~ { p : string, [number]: boolean } } else if (t2->indexer) { TypeId keyTy = reduce(t2->indexer->indexType); TypeId valueTy = reduce(t2->indexer->indexResultType); table->indexer = TableIndexer{keyTy, valueTy}; // { p : string } & { [number]: boolean } ~ { p : string, [number]: boolean } } return resultTy; } else if (auto [mt, tt] = get2(left, right); mt && tt) return std::nullopt; // TODO else if (auto [tt, mt] = get2(left, right); tt && mt) return intersectionType(right, left); // T & M ~ M & T else if (auto [m1, m2] = get2(left, right); m1 && m2) return std::nullopt; // TODO else if (auto [nl, nr] = get2(left, right); nl && nr) { // These should've been reduced already. TypeId nlTy = follow(nl->ty); TypeId nrTy = follow(nr->ty); LUAU_ASSERT(!get(nlTy) && !get(nrTy)); LUAU_ASSERT(!get(nlTy) && !get(nrTy)); LUAU_ASSERT(!get(nlTy) && !get(nrTy)); LUAU_ASSERT(!get(nlTy) && !get(nrTy)); LUAU_ASSERT(!get(nlTy) && !get(nrTy)); if (auto [npl, npr] = get2(nlTy, nrTy); npl && npr) { if (npl->type == npr->type) return left; // ~P1 & ~P2 ~ ~P1 iff P1 == P2 else return std::nullopt; // ~P1 & ~P2 ~ ~P1 & ~P2 iff P1 != P2 } else if (auto [nsl, nsr] = get2(nlTy, nrTy); nsl && nsr) { if (*nsl == *nsr) return left; // ~"A" & ~"A" ~ ~"A" else return std::nullopt; // ~"A" & ~"B" ~ ~"A" & ~"B" } else if (auto [ns, np] = get2(nlTy, nrTy); ns && np) { if (get(ns) && np->type == PrimitiveType::String) return right; // ~"A" & ~string ~ ~string else if (get(ns) && np->type == PrimitiveType::Boolean) return right; // ~false & ~boolean ~ ~boolean else return std::nullopt; // ~"A" | ~P ~ ~"A" & ~P } else if (auto [np, ns] = get2(nlTy, nrTy); np && ns) return intersectionType(right, left); // ~P & ~S ~ ~S & ~P else return std::nullopt; // ~T & ~U ~ ~T & ~U } else if (auto nl = get(left)) { // These should've been reduced already. TypeId nlTy = follow(nl->ty); LUAU_ASSERT(!get(nlTy)); LUAU_ASSERT(!get(nlTy)); LUAU_ASSERT(!get(nlTy)); LUAU_ASSERT(!get(nlTy)); LUAU_ASSERT(!get(nlTy)); if (auto [np, p] = get2(nlTy, right); np && p) { if (np->type == p->type) return builtinTypes->neverType; // ~P1 & P2 ~ never iff P1 == P2 else return right; // ~P1 & P2 ~ P2 iff P1 != P2 } else if (auto [ns, s] = get2(nlTy, right); ns && s) { if (*ns == *s) return builtinTypes->neverType; // ~"A" & "A" ~ never else return right; // ~"A" & "B" ~ "B" } else if (auto [ns, p] = get2(nlTy, right); ns && p) { if (get(ns) && p->type == PrimitiveType::String) return std::nullopt; // ~"A" & string ~ ~"A" & string else if (get(ns) && p->type == PrimitiveType::Boolean) { // Because booleans contain a fixed amount of values (2), we can do something cooler with this one. const BooleanSingleton* b = get(ns); return arena->addType(SingletonType{BooleanSingleton{!b->value}}); // ~false & boolean ~ true } else return right; // ~"A" & number ~ number } else if (auto [np, s] = get2(nlTy, right); np && s) { if (np->type == PrimitiveType::String && get(s)) return builtinTypes->neverType; // ~string & "A" ~ never else if (np->type == PrimitiveType::Boolean && get(s)) return builtinTypes->neverType; // ~boolean & true ~ never else return right; // ~P & "A" ~ "A" } else if (auto [np, f] = get2(nlTy, right); np && f) { if (np->type == PrimitiveType::Function) return builtinTypes->neverType; // ~function & () -> () ~ never else return right; // ~string & () -> () ~ () -> () } else if (auto [nc, c] = get2(nlTy, right); nc && c) { if (isSubclass(c, nc)) return builtinTypes->neverType; // ~Base & Derived ~ never else if (isSubclass(nc, c)) return std::nullopt; // ~Derived & Base ~ ~Derived & Base else return right; // ~Base & Unrelated ~ Unrelated } else if (auto [np, t] = get2(nlTy, right); np && t) { if (np->type == PrimitiveType::Table) return builtinTypes->neverType; // ~table & {} ~ never else return right; // ~string & {} ~ {} } else if (auto [np, t] = get2(nlTy, right); np && t) { if (np->type == PrimitiveType::Table) return builtinTypes->neverType; // ~table & {} ~ never else return right; // ~string & {} ~ {} } else return right; // ~T & U ~ U } else if (get(right)) return intersectionType(right, left); // T & ~U ~ ~U & T else return builtinTypes->neverType; // for all T and U except the ones handled above, T & U ~ never } std::optional TypeReducer::unionType(TypeId left, TypeId right) { LUAU_ASSERT(!get(left)); LUAU_ASSERT(!get(right)); if (get(left)) return right; // never | T ~ T else if (get(right)) return left; // T | never ~ T else if (get(left)) return left; // unknown | T ~ unknown else if (get(right)) return right; // T | unknown ~ unknown else if (get(left)) return left; // any | T ~ any else if (get(right)) return right; // T | any ~ any else if (get(left)) return std::nullopt; // error | T ~ error | T else if (get(right)) return std::nullopt; // T | error ~ T | error else if (auto [p1, p2] = get2(left, right); p1 && p2) { if (p1->type == p2->type) return left; // P1 | P2 ~ P1 iff P1 == P2 else return std::nullopt; // P1 | P2 ~ P1 | P2 iff P1 != P2 } else if (auto [p, s] = get2(left, right); p && s) { if (p->type == PrimitiveType::String && get(s)) return left; // string | "A" ~ string else if (p->type == PrimitiveType::Boolean && get(s)) return left; // boolean | true ~ boolean else return std::nullopt; // string | true ~ string | true } else if (auto [s, p] = get2(left, right); s && p) return unionType(right, left); // S | P ~ P | S else if (auto [p, f] = get2(left, right); p && f) { if (p->type == PrimitiveType::Function) return left; // function | () -> () ~ function else return std::nullopt; // P | () -> () ~ P | () -> () } else if (auto [f, p] = get2(left, right); f && p) return unionType(right, left); // () -> () | P ~ P | () -> () else if (auto [p, t] = get2(left, right); p && t) { if (p->type == PrimitiveType::Table) return left; // table | {} ~ table else return std::nullopt; // P | {} ~ P | {} } else if (auto [p, t] = get2(left, right); p && t) { if (p->type == PrimitiveType::Table) return left; // table | {} ~ table else return std::nullopt; // P | {} ~ P | {} } else if (auto [t, p] = get2(left, right); t && p) return unionType(right, left); // {} | P ~ P | {} else if (auto [t, p] = get2(left, right); t && p) return unionType(right, left); // M | P ~ P | M else if (auto [s1, s2] = get2(left, right); s1 && s2) { if (*s1 == *s2) return left; // "a" | "a" ~ "a" else return std::nullopt; // "a" | "b" ~ "a" | "b" } else if (auto [c1, c2] = get2(left, right); c1 && c2) { if (isSubclass(c1, c2)) return right; // Derived | Base ~ Base else if (isSubclass(c2, c1)) return left; // Base | Derived ~ Base else return std::nullopt; // Base | Unrelated ~ Base | Unrelated } else if (auto [nt, it] = get2(left, right); nt && it) return reduce(distribute(begin(it), end(it), &TypeReducer::unionType, left)); // ~T | (A & B) ~ (~T | A) & (~T | B) else if (auto [it, nt] = get2(left, right); it && nt) return unionType(right, left); // (A & B) | ~T ~ ~T | (A & B) else if (auto it = get(left)) { bool didReduce = false; std::vector parts; for (TypeId part : it) { auto nt = get(part); if (!nt) { parts.push_back(part); continue; } auto redex = unionType(part, right); if (redex && get(*redex)) { didReduce = true; continue; } parts.push_back(part); } if (didReduce) return flatten(std::move(parts)); // (T & ~nil) | nil ~ T else return std::nullopt; // (T & ~nil) | U } else if (get(right)) return unionType(right, left); // A | (T & U) ~ (T & U) | A else if (auto [nl, nr] = get2(left, right); nl && nr) { // These should've been reduced already. TypeId nlTy = follow(nl->ty); TypeId nrTy = follow(nr->ty); LUAU_ASSERT(!get(nlTy) && !get(nrTy)); LUAU_ASSERT(!get(nlTy) && !get(nrTy)); LUAU_ASSERT(!get(nlTy) && !get(nrTy)); LUAU_ASSERT(!get(nlTy) && !get(nrTy)); LUAU_ASSERT(!get(nlTy) && !get(nrTy)); if (auto [npl, npr] = get2(nlTy, nrTy); npl && npr) { if (npl->type == npr->type) return left; // ~P1 | ~P2 ~ ~P1 iff P1 == P2 else return builtinTypes->unknownType; // ~P1 | ~P2 ~ ~P1 iff P1 != P2 } else if (auto [nsl, nsr] = get2(nlTy, nrTy); nsl && nsr) { if (*nsl == *nsr) return left; // ~"A" | ~"A" ~ ~"A" else return builtinTypes->unknownType; // ~"A" | ~"B" ~ unknown } else if (auto [ns, np] = get2(nlTy, nrTy); ns && np) { if (get(ns) && np->type == PrimitiveType::String) return left; // ~"A" | ~string ~ ~"A" else if (get(ns) && np->type == PrimitiveType::Boolean) return left; // ~false | ~boolean ~ ~false else return builtinTypes->unknownType; // ~"A" | ~P ~ unknown } else if (auto [np, ns] = get2(nlTy, nrTy); np && ns) return unionType(right, left); // ~P | ~S ~ ~S | ~P else return std::nullopt; // TODO! } else if (auto nl = get(left)) { // These should've been reduced already. TypeId nlTy = follow(nl->ty); LUAU_ASSERT(!get(nlTy)); LUAU_ASSERT(!get(nlTy)); LUAU_ASSERT(!get(nlTy)); LUAU_ASSERT(!get(nlTy)); LUAU_ASSERT(!get(nlTy)); if (auto [np, p] = get2(nlTy, right); np && p) { if (np->type == p->type) return builtinTypes->unknownType; // ~P1 | P2 ~ unknown iff P1 == P2 else return left; // ~P1 | P2 ~ ~P1 iff P1 != P2 } else if (auto [ns, s] = get2(nlTy, right); ns && s) { if (*ns == *s) return builtinTypes->unknownType; // ~"A" | "A" ~ unknown else return left; // ~"A" | "B" ~ ~"A" } else if (auto [ns, p] = get2(nlTy, right); ns && p) { if (get(ns) && p->type == PrimitiveType::String) return builtinTypes->unknownType; // ~"A" | string ~ unknown else if (get(ns) && p->type == PrimitiveType::Boolean) return builtinTypes->unknownType; // ~false | boolean ~ unknown else return left; // ~"A" | T ~ ~"A" } else if (auto [np, s] = get2(nlTy, right); np && s) { if (np->type == PrimitiveType::String && get(s)) return std::nullopt; // ~string | "A" ~ ~string | "A" else if (np->type == PrimitiveType::Boolean && get(s)) { const BooleanSingleton* b = get(s); return negationType(arena->addType(SingletonType{BooleanSingleton{!b->value}})); // ~boolean | false ~ ~true } else return left; // ~P | "A" ~ ~P } else if (auto [nc, c] = get2(nlTy, right); nc && c) { if (isSubclass(c, nc)) return std::nullopt; // ~Base | Derived ~ ~Base | Derived else if (isSubclass(nc, c)) return builtinTypes->unknownType; // ~Derived | Base ~ unknown else return left; // ~Base | Unrelated ~ ~Base } else if (auto [np, t] = get2(nlTy, right); np && t) { if (np->type == PrimitiveType::Table) return std::nullopt; // ~table | {} ~ ~table | {} else return right; // ~P | {} ~ ~P | {} } else if (auto [np, t] = get2(nlTy, right); np && t) { if (np->type == PrimitiveType::Table) return std::nullopt; // ~table | {} ~ ~table | {} else return right; // ~P | M ~ ~P | M } else return std::nullopt; // TODO } else if (get(right)) return unionType(right, left); // T | ~U ~ ~U | T else return std::nullopt; // for all T and U except the ones handled above, T | U ~ T | U } TypeId TypeReducer::tableType(TypeId ty) { if (auto mt = get(ty)) { auto [copiedTy, copied] = copy(ty, mt); copied->table = reduce(mt->table); copied->metatable = reduce(mt->metatable); return copiedTy; } else if (auto tt = get(ty)) { // Because of `typeof()`, we need to preserve pointer identity of free/unsealed tables so that // all mutations that occurs on this will be applied without leaking the implementation details. // As a result, we'll just use the type instead of cloning it if it's free/unsealed. // // We could choose to do in-place reductions here, but to be on the safer side, I propose that we do not. if (tt->state == TableState::Free || tt->state == TableState::Unsealed) return ty; auto [copiedTy, copied] = copy(ty, tt); for (auto& [name, prop] : copied->props) { TypeId propTy = reduce(prop.type); if (get(propTy)) return builtinTypes->neverType; else prop.type = propTy; } if (copied->indexer) { TypeId keyTy = reduce(copied->indexer->indexType); TypeId valueTy = reduce(copied->indexer->indexResultType); copied->indexer = TableIndexer{keyTy, valueTy}; } for (TypeId& ty : copied->instantiatedTypeParams) ty = reduce(ty); for (TypePackId& tp : copied->instantiatedTypePackParams) tp = reduce(tp); return copiedTy; } else handle->ice("TypeReducer::tableType expects a TableType or MetatableType"); } TypeId TypeReducer::functionType(TypeId ty) { const FunctionType* f = get(ty); if (!f) handle->ice("TypeReducer::functionType expects a FunctionType"); // TODO: once we have bounded quantification, we need to be able to reduce the generic bounds. auto [copiedTy, copied] = copy(ty, f); copied->argTypes = reduce(f->argTypes); copied->retTypes = reduce(f->retTypes); return copiedTy; } TypeId TypeReducer::negationType(TypeId ty) { const NegationType* n = get(ty); if (!n) return arena->addType(NegationType{ty}); TypeId negatedTy = follow(n->ty); if (auto nn = get(negatedTy)) return nn->ty; // ~~T ~ T else if (get(negatedTy)) return builtinTypes->unknownType; // ~never ~ unknown else if (get(negatedTy)) return builtinTypes->neverType; // ~unknown ~ never else if (get(negatedTy)) return builtinTypes->anyType; // ~any ~ any else if (auto ni = get(negatedTy)) { std::vector options; for (TypeId part : ni) options.push_back(negationType(arena->addType(NegationType{part}))); return reduce(flatten(std::move(options))); // ~(T & U) ~ (~T | ~U) } else if (auto nu = get(negatedTy)) { std::vector parts; for (TypeId option : nu) parts.push_back(negationType(arena->addType(NegationType{option}))); return reduce(flatten(std::move(parts))); // ~(T | U) ~ (~T & ~U) } else return ty; // for all T except the ones handled above, ~T ~ ~T } struct MarkCycles : TypeVisitor { DenseHashSet cyclics{nullptr}; void cycle(TypeId ty) override { cyclics.insert(follow(ty)); } void cycle(TypePackId tp) override { cyclics.insert(follow(tp)); } bool visit(TypeId ty) override { return !cyclics.find(follow(ty)); } bool visit(TypePackId tp) override { return !cyclics.find(follow(tp)); } }; } // namespace TypeReduction::TypeReduction( NotNull arena, NotNull builtinTypes, NotNull handle, const TypeReductionOptions& opts) : arena(arena) , builtinTypes(builtinTypes) , handle(handle) , options(opts) { } std::optional TypeReduction::reduce(TypeId ty) { ty = follow(ty); if (FFlag::DebugLuauDontReduceTypes) return ty; else if (!options.allowTypeReductionsFromOtherArenas && ty->owningArena != arena) return ty; else if (auto edge = memoization.memoizedof(ty)) { if (edge->irreducible) return edge->type; else ty = edge->type; } else if (hasExceededCartesianProductLimit(ty)) return std::nullopt; try { MarkCycles finder; finder.traverse(ty); TypeReducer reducer{arena, builtinTypes, handle, NotNull{&memoization}, &finder.cyclics}; return reducer.reduce(ty); } catch (const RecursionLimitException&) { return std::nullopt; } } std::optional TypeReduction::reduce(TypePackId tp) { tp = follow(tp); if (FFlag::DebugLuauDontReduceTypes) return tp; else if (!options.allowTypeReductionsFromOtherArenas && tp->owningArena != arena) return tp; else if (auto edge = memoization.memoizedof(tp)) { if (edge->irreducible) return edge->type; else tp = edge->type; } else if (hasExceededCartesianProductLimit(tp)) return std::nullopt; try { MarkCycles finder; finder.traverse(tp); TypeReducer reducer{arena, builtinTypes, handle, NotNull{&memoization}, &finder.cyclics}; return reducer.reduce(tp); } catch (const RecursionLimitException&) { return std::nullopt; } } std::optional TypeReduction::reduce(const TypeFun& fun) { if (FFlag::DebugLuauDontReduceTypes) return fun; // TODO: once we have bounded quantification, we need to be able to reduce the generic bounds. if (auto reducedTy = reduce(fun.type)) return TypeFun{fun.typeParams, fun.typePackParams, *reducedTy}; return std::nullopt; } size_t TypeReduction::cartesianProductSize(TypeId ty) const { ty = follow(ty); auto it = get(follow(ty)); if (!it) return 1; return std::accumulate(begin(it), end(it), size_t(1), [](size_t acc, TypeId ty) { if (auto ut = get(ty)) return acc * std::distance(begin(ut), end(ut)); else if (get(ty)) return acc * 0; else return acc * 1; }); } bool TypeReduction::hasExceededCartesianProductLimit(TypeId ty) const { return cartesianProductSize(ty) >= size_t(FInt::LuauTypeReductionCartesianProductLimit); } bool TypeReduction::hasExceededCartesianProductLimit(TypePackId tp) const { TypePackIterator it = begin(tp); while (it != end(tp)) { if (hasExceededCartesianProductLimit(*it)) return true; ++it; } if (auto tail = it.tail()) { if (auto vtp = get(follow(*tail))) { if (hasExceededCartesianProductLimit(vtp->ty)) return true; } } return false; } } // namespace Luau