// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details #include "Luau/Subtyping.h" #include "Luau/Common.h" #include "Luau/Error.h" #include "Luau/Normalize.h" #include "Luau/Scope.h" #include "Luau/StringUtils.h" #include "Luau/ToString.h" #include "Luau/Type.h" #include "Luau/TypeArena.h" #include "Luau/TypePack.h" #include "Luau/TypeUtils.h" #include namespace Luau { struct VarianceFlipper { Subtyping::Variance* variance; Subtyping::Variance oldValue; VarianceFlipper(Subtyping::Variance* v) : variance(v) , oldValue(*v) { switch (oldValue) { case Subtyping::Variance::Covariant: *variance = Subtyping::Variance::Contravariant; break; case Subtyping::Variance::Contravariant: *variance = Subtyping::Variance::Covariant; break; } } ~VarianceFlipper() { *variance = oldValue; } }; SubtypingResult& SubtypingResult::andAlso(const SubtypingResult& other) { isSubtype &= other.isSubtype; // `|=` is intentional here, we want to preserve error related flags. isErrorSuppressing |= other.isErrorSuppressing; normalizationTooComplex |= other.normalizationTooComplex; return *this; } SubtypingResult& SubtypingResult::orElse(const SubtypingResult& other) { isSubtype |= other.isSubtype; isErrorSuppressing |= other.isErrorSuppressing; normalizationTooComplex |= other.normalizationTooComplex; return *this; } SubtypingResult SubtypingResult::negate(const SubtypingResult& result) { return SubtypingResult{ !result.isSubtype, result.isErrorSuppressing, result.normalizationTooComplex, }; } SubtypingResult SubtypingResult::all(const std::vector& results) { SubtypingResult acc{true, false}; for (const SubtypingResult& current : results) acc.andAlso(current); return acc; } SubtypingResult SubtypingResult::any(const std::vector& results) { SubtypingResult acc{false, false}; for (const SubtypingResult& current : results) acc.orElse(current); return acc; } SubtypingResult Subtyping::isSubtype(TypeId subTy, TypeId superTy) { mappedGenerics.clear(); mappedGenericPacks.clear(); SubtypingResult result = isCovariantWith(subTy, superTy); for (const auto& [subTy, bounds] : mappedGenerics) { const auto& lb = bounds.lowerBound; const auto& ub = bounds.upperBound; TypeId lowerBound = makeAggregateType(lb, builtinTypes->neverType); TypeId upperBound = makeAggregateType(ub, builtinTypes->unknownType); result.andAlso(isCovariantWith(lowerBound, upperBound)); } return result; } SubtypingResult Subtyping::isSubtype(TypePackId subTp, TypePackId superTp) { return isCovariantWith(subTp, superTp); } namespace { struct SeenSetPopper { Subtyping::SeenSet* seenTypes; std::pair pair; SeenSetPopper(Subtyping::SeenSet* seenTypes, std::pair pair) : seenTypes(seenTypes) , pair(pair) { } ~SeenSetPopper() { seenTypes->erase(pair); } }; } // namespace SubtypingResult Subtyping::isCovariantWith(TypeId subTy, TypeId superTy) { subTy = follow(subTy); superTy = follow(superTy); // TODO: Do we care about returning a proof that this is error-suppressing? // e.g. given `a | error <: a | error` where both operands are pointer equal, // then should it also carry the information that it's error-suppressing? // If it should, then `error <: error` should also do the same. if (subTy == superTy) return {true}; std::pair typePair{subTy, superTy}; if (!seenTypes.insert(typePair).second) return {true}; SeenSetPopper ssp{&seenTypes, typePair}; // Within the scope to which a generic belongs, that generic should be // tested as though it were its upper bounds. We do not yet support bounded // generics, so the upper bound is always unknown. if (auto subGeneric = get(subTy); subGeneric && subsumes(subGeneric->scope, scope)) return isCovariantWith(builtinTypes->unknownType, superTy); if (auto superGeneric = get(superTy); superGeneric && subsumes(superGeneric->scope, scope)) return isCovariantWith(subTy, builtinTypes->unknownType); if (auto subUnion = get(subTy)) return isCovariantWith(subUnion, superTy); else if (auto superUnion = get(superTy)) return isCovariantWith(subTy, superUnion); else if (auto superIntersection = get(superTy)) return isCovariantWith(subTy, superIntersection); else if (auto subIntersection = get(subTy)) { SubtypingResult result = isCovariantWith(subIntersection, superTy); if (result.isSubtype || result.isErrorSuppressing || result.normalizationTooComplex) return result; else return isCovariantWith(normalizer->normalize(subTy), normalizer->normalize(superTy)); } else if (get(superTy)) return {true}; // This is always true. else if (get(subTy)) { // any = unknown | error, so we rewrite this to match. // As per TAPL: A | B <: T iff A <: T && B <: T return isCovariantWith(builtinTypes->unknownType, superTy).andAlso(isCovariantWith(builtinTypes->errorType, superTy)); } else if (get(superTy)) { LUAU_ASSERT(!get(subTy)); // TODO: replace with ice. LUAU_ASSERT(!get(subTy)); // TODO: replace with ice. LUAU_ASSERT(!get(subTy)); // TODO: replace with ice. bool errorSuppressing = get(subTy); return {!errorSuppressing, errorSuppressing}; } else if (get(subTy)) return {true}; else if (get(superTy)) return {false, true}; else if (get(subTy)) return {false, true}; else if (auto p = get2(subTy, superTy)) return isCovariantWith(p.first->ty, p.second->ty); else if (auto subNegation = get(subTy)) return isCovariantWith(subNegation, superTy); else if (auto superNegation = get(superTy)) return isCovariantWith(subTy, superNegation); else if (auto subGeneric = get(subTy); subGeneric && variance == Variance::Covariant) { bool ok = bindGeneric(subTy, superTy); return {ok}; } else if (auto superGeneric = get(superTy); superGeneric && variance == Variance::Contravariant) { bool ok = bindGeneric(subTy, superTy); return {ok}; } else if (auto p = get2(subTy, superTy)) return isCovariantWith(p); else if (auto p = get2(subTy, superTy)) return isCovariantWith(p); else if (auto p = get2(subTy, superTy)) return isCovariantWith(p); else if (auto p = get2(subTy, superTy)) return isCovariantWith(p); else if (auto p = get2(subTy, superTy)) return isCovariantWith(p); else if (auto p = get2(subTy, superTy)) return isCovariantWith(p); else if (auto p = get2(subTy, superTy)) return isCovariantWith(p); else if (auto p = get2(subTy, superTy)) return isCovariantWith(p); else if (auto p = get2(subTy, superTy)) return isCovariantWith(p); else if (auto p = get2(subTy, superTy)) return isCovariantWith(p); else if (auto p = get2(subTy, superTy)) return isCovariantWith(p); return {false}; } SubtypingResult Subtyping::isCovariantWith(TypePackId subTp, TypePackId superTp) { subTp = follow(subTp); superTp = follow(superTp); auto [subHead, subTail] = flatten(subTp); auto [superHead, superTail] = flatten(superTp); const size_t headSize = std::min(subHead.size(), superHead.size()); std::vector results; results.reserve(std::max(subHead.size(), superHead.size()) + 1); if (subTp == superTp) return {true}; // Match head types pairwise for (size_t i = 0; i < headSize; ++i) { results.push_back(isCovariantWith(subHead[i], superHead[i])); if (!results.back().isSubtype) return {false}; } // Handle mismatched head sizes if (subHead.size() < superHead.size()) { if (subTail) { if (auto vt = get(*subTail)) { for (size_t i = headSize; i < superHead.size(); ++i) results.push_back(isCovariantWith(vt->ty, superHead[i])); } else if (auto gt = get(*subTail)) { if (variance == Variance::Covariant) { // For any non-generic type T: // // (X) -> () <: (T) -> () // Possible optimization: If headSize == 0 then we can just use subTp as-is. std::vector headSlice(begin(superHead), begin(superHead) + headSize); TypePackId superTailPack = arena->addTypePack(std::move(headSlice), superTail); if (TypePackId* other = mappedGenericPacks.find(*subTail)) results.push_back(isCovariantWith(*other, superTailPack)); else mappedGenericPacks.try_insert(*subTail, superTailPack); // FIXME? Not a fan of the early return here. It makes the // control flow harder to reason about. return SubtypingResult::all(results); } else { // For any non-generic type T: // // (T) -> () (X) -> () // return {false}; } } else unexpected(*subTail); } else return {false}; } else if (subHead.size() > superHead.size()) { if (superTail) { if (auto vt = get(*superTail)) { for (size_t i = headSize; i < subHead.size(); ++i) results.push_back(isCovariantWith(subHead[i], vt->ty)); } else if (auto gt = get(*superTail)) { if (variance == Variance::Contravariant) { // For any non-generic type T: // // (X...) -> () <: (T) -> () // Possible optimization: If headSize == 0 then we can just use subTp as-is. std::vector headSlice(begin(subHead), begin(subHead) + headSize); TypePackId subTailPack = arena->addTypePack(std::move(headSlice), subTail); if (TypePackId* other = mappedGenericPacks.find(*superTail)) results.push_back(isCovariantWith(*other, subTailPack)); else mappedGenericPacks.try_insert(*superTail, subTailPack); // FIXME? Not a fan of the early return here. It makes the // control flow harder to reason about. return SubtypingResult::all(results); } else { // For any non-generic type T: // // () -> T () -> X... return {false}; } } else unexpected(*superTail); } else return {false}; } // Handle tails if (subTail && superTail) { if (auto p = get2(*subTail, *superTail)) { results.push_back(isCovariantWith(p)); } else if (auto p = get2(*subTail, *superTail)) { bool ok = bindGeneric(*subTail, *superTail); results.push_back({ok}); } else if (get2(*subTail, *superTail)) { if (variance == Variance::Contravariant) { // (A...) -> number <: (...number) -> number bool ok = bindGeneric(*subTail, *superTail); results.push_back({ok}); } else { // (number) -> ...number (number) -> A... results.push_back({false}); } } else if (get2(*subTail, *superTail)) { if (variance == Variance::Contravariant) { // (...number) -> number (A...) -> number results.push_back({false}); } else { // () -> A... <: () -> ...number bool ok = bindGeneric(*subTail, *superTail); results.push_back({ok}); } } else iceReporter->ice( format("Subtyping::isSubtype got unexpected type packs %s and %s", toString(*subTail).c_str(), toString(*superTail).c_str())); } else if (subTail) { if (get(*subTail)) { return {false}; } else if (get(*subTail)) { bool ok = bindGeneric(*subTail, builtinTypes->emptyTypePack); return {ok}; } else unexpected(*subTail); } else if (superTail) { if (get(*superTail)) { /* * A variadic type pack ...T can be thought of as an infinite union of finite type packs. * () | (T) | (T, T) | (T, T, T) | ... * * And, per TAPL: * T <: A | B iff T <: A or T <: B * * All variadic type packs are therefore supertypes of the empty type pack. */ } else if (get(*superTail)) { if (variance == Variance::Contravariant) { bool ok = bindGeneric(builtinTypes->emptyTypePack, *superTail); results.push_back({ok}); } else results.push_back({false}); } else LUAU_ASSERT(0); // TODO } return SubtypingResult::all(results); } template SubtypingResult Subtyping::isContravariantWith(SubTy&& subTy, SuperTy&& superTy) { return isCovariantWith(superTy, subTy); } template SubtypingResult Subtyping::isInvariantWith(SubTy&& subTy, SuperTy&& superTy) { return isCovariantWith(subTy, superTy).andAlso(isContravariantWith(subTy, superTy)); } template SubtypingResult Subtyping::isCovariantWith(const TryPair& pair) { return isCovariantWith(pair.first, pair.second); } template SubtypingResult Subtyping::isContravariantWith(const TryPair& pair) { return isCovariantWith(pair.second, pair.first); } template SubtypingResult Subtyping::isInvariantWith(const TryPair& pair) { return isCovariantWith(pair).andAlso(isContravariantWith(pair)); } /* * This is much simpler than the Unifier implementation because we don't * actually care about potential "cross-talk" between union parts that match the * left side. * * In fact, we're very limited in what we can do: If multiple choices match, but * all of them have non-overlapping constraints, then we're stuck with an "or" * conjunction of constraints. Solving this in the general case is quite * difficult. * * For example, we cannot dispatch anything from this constraint: * * {x: number, y: string} <: {x: number, y: 'a} | {x: 'b, y: string} * * From this constraint, we can know that either string <: 'a or number <: 'b, * but we don't know which! * * However: * * {x: number, y: string} <: {x: number, y: 'a} | {x: number, y: string} * * We can dispatch this constraint because there is no 'or' conjunction. One of * the arms requires 0 matches. * * {x: number, y: string, z: boolean} | {x: number, y: 'a, z: 'b} | {x: number, * y: string, z: 'b} * * Here, we have two matches. One asks for string ~ 'a and boolean ~ 'b. The * other just asks for boolean ~ 'b. We can dispatch this and only commit * boolean ~ 'b. This constraint does not teach us anything about 'a. */ SubtypingResult Subtyping::isCovariantWith(TypeId subTy, const UnionType* superUnion) { // As per TAPL: T <: A | B iff T <: A || T <: B std::vector subtypings; for (TypeId ty : superUnion) subtypings.push_back(isCovariantWith(subTy, ty)); return SubtypingResult::any(subtypings); } SubtypingResult Subtyping::isCovariantWith(const UnionType* subUnion, TypeId superTy) { // As per TAPL: A | B <: T iff A <: T && B <: T std::vector subtypings; for (TypeId ty : subUnion) subtypings.push_back(isCovariantWith(ty, superTy)); return SubtypingResult::all(subtypings); } SubtypingResult Subtyping::isCovariantWith(TypeId subTy, const IntersectionType* superIntersection) { // As per TAPL: T <: A & B iff T <: A && T <: B std::vector subtypings; for (TypeId ty : superIntersection) subtypings.push_back(isCovariantWith(subTy, ty)); return SubtypingResult::all(subtypings); } SubtypingResult Subtyping::isCovariantWith(const IntersectionType* subIntersection, TypeId superTy) { // As per TAPL: A & B <: T iff A <: T || B <: T std::vector subtypings; for (TypeId ty : subIntersection) subtypings.push_back(isCovariantWith(ty, superTy)); return SubtypingResult::any(subtypings); } SubtypingResult Subtyping::isCovariantWith(const NegationType* subNegation, TypeId superTy) { TypeId negatedTy = follow(subNegation->ty); // In order to follow a consistent codepath, rather than folding the // isCovariantWith test down to its conclusion here, we test the subtyping test // of the result of negating the type for never, unknown, any, and error. if (is(negatedTy)) { // ¬never ~ unknown return isCovariantWith(builtinTypes->unknownType, superTy); } else if (is(negatedTy)) { // ¬unknown ~ never return isCovariantWith(builtinTypes->neverType, superTy); } else if (is(negatedTy)) { // ¬any ~ any return isCovariantWith(negatedTy, superTy); } else if (auto u = get(negatedTy)) { // ¬(A ∪ B) ~ ¬A ∩ ¬B // follow intersection rules: A & B <: T iff A <: T && B <: T std::vector subtypings; for (TypeId ty : u) { NegationType negatedTmp{ty}; subtypings.push_back(isCovariantWith(&negatedTmp, superTy)); } return SubtypingResult::all(subtypings); } else if (auto i = get(negatedTy)) { // ¬(A ∩ B) ~ ¬A ∪ ¬B // follow union rules: A | B <: T iff A <: T || B <: T std::vector subtypings; for (TypeId ty : i) { if (auto negatedPart = get(follow(ty))) subtypings.push_back(isCovariantWith(negatedPart->ty, superTy)); else { NegationType negatedTmp{ty}; subtypings.push_back(isCovariantWith(&negatedTmp, superTy)); } } return SubtypingResult::any(subtypings); } else if (is(negatedTy)) { iceReporter->ice("attempting to negate a non-testable type"); } // negating a different subtype will get you a very wide type that's not a // subtype of other stuff. else { return {false}; } } SubtypingResult Subtyping::isCovariantWith(const TypeId subTy, const NegationType* superNegation) { TypeId negatedTy = follow(superNegation->ty); if (is(negatedTy)) { // ¬never ~ unknown return isCovariantWith(subTy, builtinTypes->unknownType); } else if (is(negatedTy)) { // ¬unknown ~ never return isCovariantWith(subTy, builtinTypes->neverType); } else if (is(negatedTy)) { // ¬any ~ any return isSubtype(subTy, negatedTy); } else if (auto u = get(negatedTy)) { // ¬(A ∪ B) ~ ¬A ∩ ¬B // follow intersection rules: A & B <: T iff A <: T && B <: T std::vector subtypings; for (TypeId ty : u) { if (auto negatedPart = get(follow(ty))) subtypings.push_back(isCovariantWith(subTy, negatedPart->ty)); else { NegationType negatedTmp{ty}; subtypings.push_back(isCovariantWith(subTy, &negatedTmp)); } } return SubtypingResult::all(subtypings); } else if (auto i = get(negatedTy)) { // ¬(A ∩ B) ~ ¬A ∪ ¬B // follow union rules: A | B <: T iff A <: T || B <: T std::vector subtypings; for (TypeId ty : i) { if (auto negatedPart = get(follow(ty))) subtypings.push_back(isCovariantWith(subTy, negatedPart->ty)); else { NegationType negatedTmp{ty}; subtypings.push_back(isCovariantWith(subTy, &negatedTmp)); } } return SubtypingResult::any(subtypings); } else if (auto p = get2(subTy, negatedTy)) { // number <: ¬boolean // number type != p.second->type}; } else if (auto p = get2(subTy, negatedTy)) { // "foo" (p.first) && p.second->type == PrimitiveType::String) return {false}; // false (p.first) && p.second->type == PrimitiveType::Boolean) return {false}; // other cases are true else return {true}; } else if (auto p = get2(subTy, negatedTy)) { if (p.first->type == PrimitiveType::String && get(p.second)) return {false}; else if (p.first->type == PrimitiveType::Boolean && get(p.second)) return {false}; else return {true}; } // the top class type is not actually a primitive type, so the negation of // any one of them includes the top class type. else if (auto p = get2(subTy, negatedTy)) return {true}; else if (auto p = get(negatedTy); p && is(subTy)) return {p->type != PrimitiveType::Table}; else if (auto p = get2(subTy, negatedTy)) return {p.second->type != PrimitiveType::Function}; else if (auto p = get2(subTy, negatedTy)) return {*p.first != *p.second}; else if (auto p = get2(subTy, negatedTy)) return SubtypingResult::negate(isCovariantWith(p.first, p.second)); else if (get2(subTy, negatedTy)) return {true}; else if (is(negatedTy)) iceReporter->ice("attempting to negate a non-testable type"); return {false}; } SubtypingResult Subtyping::isCovariantWith(const PrimitiveType* subPrim, const PrimitiveType* superPrim) { return {subPrim->type == superPrim->type}; } SubtypingResult Subtyping::isCovariantWith(const SingletonType* subSingleton, const PrimitiveType* superPrim) { if (get(subSingleton) && superPrim->type == PrimitiveType::String) return {true}; else if (get(subSingleton) && superPrim->type == PrimitiveType::Boolean) return {true}; else return {false}; } SubtypingResult Subtyping::isCovariantWith(const SingletonType* subSingleton, const SingletonType* superSingleton) { return {*subSingleton == *superSingleton}; } SubtypingResult Subtyping::isCovariantWith(const TableType* subTable, const TableType* superTable) { SubtypingResult result{true}; for (const auto& [name, prop] : superTable->props) { auto it = subTable->props.find(name); if (it != subTable->props.end()) result.andAlso(isInvariantWith(prop.type(), it->second.type())); else return SubtypingResult{false}; } return result; } SubtypingResult Subtyping::isCovariantWith(const MetatableType* subMt, const MetatableType* superMt) { return SubtypingResult::all({ isCovariantWith(subMt->table, superMt->table), isCovariantWith(subMt->metatable, superMt->metatable), }); } SubtypingResult Subtyping::isCovariantWith(const MetatableType* subMt, const TableType* superTable) { if (auto subTable = get(subMt->table)) { // Metatables cannot erase properties from the table they're attached to, so // the subtyping rule for this is just if the table component is a subtype // of the supertype table. // // There's a flaw here in that if the __index metamethod contributes a new // field that would satisfy the subtyping relationship, we'll erronously say // that the metatable isn't a subtype of the table, even though they have // compatible properties/shapes. We'll revisit this later when we have a // better understanding of how important this is. return isCovariantWith(subTable, superTable); } else { // TODO: This may be a case we actually hit? return {false}; } } SubtypingResult Subtyping::isCovariantWith(const ClassType* subClass, const ClassType* superClass) { return {isSubclass(subClass, superClass)}; } SubtypingResult Subtyping::isCovariantWith(const ClassType* subClass, const TableType* superTable) { SubtypingResult result{true}; for (const auto& [name, prop] : superTable->props) { if (auto classProp = lookupClassProp(subClass, name)) result.andAlso(isInvariantWith(prop.type(), classProp->type())); else return SubtypingResult{false}; } return result; } SubtypingResult Subtyping::isCovariantWith(const FunctionType* subFunction, const FunctionType* superFunction) { SubtypingResult result; { VarianceFlipper vf{&variance}; result.orElse(isContravariantWith(subFunction->argTypes, superFunction->argTypes)); } result.andAlso(isCovariantWith(subFunction->retTypes, superFunction->retTypes)); return result; } SubtypingResult Subtyping::isCovariantWith(const PrimitiveType* subPrim, const TableType* superTable) { SubtypingResult result{false}; if (subPrim->type == PrimitiveType::String) { if (auto metatable = getMetatable(builtinTypes->stringType, builtinTypes)) { if (auto mttv = get(follow(metatable))) { if (auto it = mttv->props.find("__index"); it != mttv->props.end()) { if (auto stringTable = get(it->second.type())) result.orElse(isCovariantWith(stringTable, superTable)); } } } } return result; } SubtypingResult Subtyping::isCovariantWith(const SingletonType* subSingleton, const TableType* superTable) { SubtypingResult result{false}; if (auto stringleton = get(subSingleton)) { if (auto metatable = getMetatable(builtinTypes->stringType, builtinTypes)) { if (auto mttv = get(follow(metatable))) { if (auto it = mttv->props.find("__index"); it != mttv->props.end()) { if (auto stringTable = get(it->second.type())) result.orElse(isCovariantWith(stringTable, superTable)); } } } } return result; } SubtypingResult Subtyping::isCovariantWith(const NormalizedType* subNorm, const NormalizedType* superNorm) { if (!subNorm || !superNorm) return {false, true, true}; SubtypingResult result = isCovariantWith(subNorm->tops, superNorm->tops); result.andAlso(isCovariantWith(subNorm->booleans, superNorm->booleans)); result.andAlso(isCovariantWith(subNorm->classes, superNorm->classes).orElse(isCovariantWith(subNorm->classes, superNorm->tables))); result.andAlso(isCovariantWith(subNorm->errors, superNorm->errors)); result.andAlso(isCovariantWith(subNorm->nils, superNorm->nils)); result.andAlso(isCovariantWith(subNorm->numbers, superNorm->numbers)); result.andAlso(isCovariantWith(subNorm->strings, superNorm->strings)); result.andAlso(isCovariantWith(subNorm->strings, superNorm->tables)); result.andAlso(isCovariantWith(subNorm->threads, superNorm->threads)); result.andAlso(isCovariantWith(subNorm->tables, superNorm->tables)); result.andAlso(isCovariantWith(subNorm->functions, superNorm->functions)); // isCovariantWith(subNorm->tyvars, superNorm->tyvars); return result; } SubtypingResult Subtyping::isCovariantWith(const NormalizedClassType& subClass, const NormalizedClassType& superClass) { for (const auto& [subClassTy, _] : subClass.classes) { SubtypingResult result; for (const auto& [superClassTy, superNegations] : superClass.classes) { result.orElse(isCovariantWith(subClassTy, superClassTy)); if (!result.isSubtype) continue; for (TypeId negation : superNegations) { result.andAlso(SubtypingResult::negate(isCovariantWith(subClassTy, negation))); if (result.isSubtype) break; } } if (!result.isSubtype) return result; } return {true}; } SubtypingResult Subtyping::isCovariantWith(const NormalizedClassType& subClass, const TypeIds& superTables) { for (const auto& [subClassTy, _] : subClass.classes) { SubtypingResult result; for (TypeId superTableTy : superTables) result.orElse(isCovariantWith(subClassTy, superTableTy)); if (!result.isSubtype) return result; } return {true}; } SubtypingResult Subtyping::isCovariantWith(const NormalizedStringType& subString, const NormalizedStringType& superString) { bool isSubtype = Luau::isSubtype(subString, superString); return {isSubtype}; } SubtypingResult Subtyping::isCovariantWith(const NormalizedStringType& subString, const TypeIds& superTables) { if (subString.isNever()) return {true}; if (subString.isCofinite) { SubtypingResult result; for (const auto& superTable : superTables) { result.orElse(isCovariantWith(builtinTypes->stringType, superTable)); if (result.isSubtype) return result; } return result; } // Finite case // S = s1 | s2 | s3 ... sn <: t1 | t2 | ... | tn // iff for some ti, S <: ti // iff for all sj, sj <: ti for (const auto& superTable : superTables) { SubtypingResult result{true}; for (const auto& [_, subString] : subString.singletons) { result.andAlso(isCovariantWith(subString, superTable)); if (!result.isSubtype) break; } if (!result.isSubtype) continue; else return result; } return {false}; } SubtypingResult Subtyping::isCovariantWith(const NormalizedFunctionType& subFunction, const NormalizedFunctionType& superFunction) { if (subFunction.isNever()) return {true}; else if (superFunction.isTop) return {true}; else return isCovariantWith(subFunction.parts, superFunction.parts); } SubtypingResult Subtyping::isCovariantWith(const TypeIds& subTypes, const TypeIds& superTypes) { std::vector results; for (TypeId subTy : subTypes) { results.emplace_back(); for (TypeId superTy : superTypes) results.back().orElse(isCovariantWith(subTy, superTy)); } return SubtypingResult::all(results); } SubtypingResult Subtyping::isCovariantWith(const VariadicTypePack* subVariadic, const VariadicTypePack* superVariadic) { return isCovariantWith(subVariadic->ty, superVariadic->ty); } bool Subtyping::bindGeneric(TypeId subTy, TypeId superTy) { if (variance == Variance::Covariant) { if (!get(subTy)) return false; mappedGenerics[subTy].upperBound.insert(superTy); } else { if (!get(superTy)) return false; mappedGenerics[superTy].lowerBound.insert(subTy); } return true; } /* * If, when performing a subtyping test, we encounter a generic on the left * side, it is permissible to tentatively bind that generic to the right side * type. */ bool Subtyping::bindGeneric(TypePackId subTp, TypePackId superTp) { if (variance == Variance::Contravariant) std::swap(superTp, subTp); if (!get(subTp)) return false; if (TypePackId* m = mappedGenericPacks.find(subTp)) return *m == superTp; mappedGenericPacks[subTp] = superTp; return true; } template TypeId Subtyping::makeAggregateType(const Container& container, TypeId orElse) { if (container.empty()) return orElse; else if (container.size() == 1) return *begin(container); else return arena->addType(T{std::vector(begin(container), end(container))}); } void Subtyping::unexpected(TypePackId tp) { iceReporter->ice(format("Unexpected type pack %s", toString(tp).c_str())); } } // namespace Luau