// 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/RecursionCounter.h" #include "Luau/Scope.h" #include "Luau/StringUtils.h" #include "Luau/Substitution.h" #include "Luau/ToString.h" #include "Luau/TxnLog.h" #include "Luau/Type.h" #include "Luau/TypeArena.h" #include "Luau/TypeCheckLimits.h" #include "Luau/TypeFunction.h" #include "Luau/TypePack.h" #include "Luau/TypePath.h" #include "Luau/TypeUtils.h" #include LUAU_FASTFLAGVARIABLE(DebugLuauSubtypingCheckPathValidity) LUAU_FASTFLAGVARIABLE(LuauSubtypingFixTailPack) 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; } }; bool SubtypingReasoning::operator==(const SubtypingReasoning& other) const { return subPath == other.subPath && superPath == other.superPath && variance == other.variance; } size_t SubtypingReasoningHash::operator()(const SubtypingReasoning& r) const { return TypePath::PathHash()(r.subPath) ^ (TypePath::PathHash()(r.superPath) << 1) ^ (static_cast(r.variance) << 1); } template static void assertReasoningValid(TID subTy, TID superTy, const SubtypingResult& result, NotNull builtinTypes) { if (!FFlag::DebugLuauSubtypingCheckPathValidity) return; for (const SubtypingReasoning& reasoning : result.reasoning) { LUAU_ASSERT(traverse(subTy, reasoning.subPath, builtinTypes)); LUAU_ASSERT(traverse(superTy, reasoning.superPath, builtinTypes)); } } template<> void assertReasoningValid(TableIndexer subIdx, TableIndexer superIdx, const SubtypingResult& result, NotNull builtinTypes) { // Empty method to satisfy the compiler. } static SubtypingReasonings mergeReasonings(const SubtypingReasonings& a, const SubtypingReasonings& b) { SubtypingReasonings result{kEmptyReasoning}; for (const SubtypingReasoning& r : a) { if (r.variance == SubtypingVariance::Invariant) result.insert(r); else if (r.variance == SubtypingVariance::Covariant || r.variance == SubtypingVariance::Contravariant) { SubtypingReasoning inverseReasoning = SubtypingReasoning{ r.subPath, r.superPath, r.variance == SubtypingVariance::Covariant ? SubtypingVariance::Contravariant : SubtypingVariance::Covariant }; if (b.contains(inverseReasoning)) result.insert(SubtypingReasoning{r.subPath, r.superPath, SubtypingVariance::Invariant}); else result.insert(r); } } for (const SubtypingReasoning& r : b) { if (r.variance == SubtypingVariance::Invariant) result.insert(r); else if (r.variance == SubtypingVariance::Covariant || r.variance == SubtypingVariance::Contravariant) { SubtypingReasoning inverseReasoning = SubtypingReasoning{ r.subPath, r.superPath, r.variance == SubtypingVariance::Covariant ? SubtypingVariance::Contravariant : SubtypingVariance::Covariant }; if (a.contains(inverseReasoning)) result.insert(SubtypingReasoning{r.subPath, r.superPath, SubtypingVariance::Invariant}); else result.insert(r); } } return result; } SubtypingResult& SubtypingResult::andAlso(const SubtypingResult& other) { // If the other result is not a subtype, we want to join all of its // reasonings to this one. If this result already has reasonings of its own, // those need to be attributed here whenever this _also_ failed. if (!other.isSubtype) reasoning = isSubtype ? std::move(other.reasoning) : mergeReasonings(reasoning, other.reasoning); isSubtype &= other.isSubtype; normalizationTooComplex |= other.normalizationTooComplex; isCacheable &= other.isCacheable; errors.insert(errors.end(), other.errors.begin(), other.errors.end()); return *this; } SubtypingResult& SubtypingResult::orElse(const SubtypingResult& other) { // If this result is a subtype, we do not join the reasoning lists. If this // result is not a subtype, but the other is a subtype, we want to _clear_ // our reasoning list. If both results are not subtypes, we join the // reasoning lists. if (!isSubtype) { if (other.isSubtype) reasoning.clear(); else reasoning = mergeReasonings(reasoning, other.reasoning); } isSubtype |= other.isSubtype; normalizationTooComplex |= other.normalizationTooComplex; isCacheable &= other.isCacheable; errors.insert(errors.end(), other.errors.begin(), other.errors.end()); return *this; } SubtypingResult& SubtypingResult::withBothComponent(TypePath::Component component) { return withSubComponent(component).withSuperComponent(component); } SubtypingResult& SubtypingResult::withSubComponent(TypePath::Component component) { if (reasoning.empty()) reasoning.insert(SubtypingReasoning{Path(component), TypePath::kEmpty}); else { for (auto& r : reasoning) r.subPath = r.subPath.push_front(component); } return *this; } SubtypingResult& SubtypingResult::withSuperComponent(TypePath::Component component) { if (reasoning.empty()) reasoning.insert(SubtypingReasoning{TypePath::kEmpty, Path(component)}); else { for (auto& r : reasoning) r.superPath = r.superPath.push_front(component); } return *this; } SubtypingResult& SubtypingResult::withBothPath(TypePath::Path path) { return withSubPath(path).withSuperPath(path); } SubtypingResult& SubtypingResult::withSubPath(TypePath::Path path) { if (reasoning.empty()) reasoning.insert(SubtypingReasoning{path, TypePath::kEmpty}); else { for (auto& r : reasoning) r.subPath = path.append(r.subPath); } return *this; } SubtypingResult& SubtypingResult::withSuperPath(TypePath::Path path) { if (reasoning.empty()) reasoning.insert(SubtypingReasoning{TypePath::kEmpty, path}); else { for (auto& r : reasoning) r.superPath = path.append(r.superPath); } return *this; } SubtypingResult& SubtypingResult::withErrors(ErrorVec& err) { for (TypeError& e : err) errors.emplace_back(e); return *this; } SubtypingResult& SubtypingResult::withError(TypeError err) { errors.push_back(std::move(err)); return *this; } SubtypingResult SubtypingResult::negate(const SubtypingResult& result) { return SubtypingResult{ !result.isSubtype, result.normalizationTooComplex, }; } SubtypingResult SubtypingResult::all(const std::vector& results) { SubtypingResult acc{true}; for (const SubtypingResult& current : results) acc.andAlso(current); return acc; } SubtypingResult SubtypingResult::any(const std::vector& results) { SubtypingResult acc{false}; for (const SubtypingResult& current : results) acc.orElse(current); return acc; } struct ApplyMappedGenerics : Substitution { NotNull builtinTypes; NotNull arena; SubtypingEnvironment& env; ApplyMappedGenerics(NotNull builtinTypes, NotNull arena, SubtypingEnvironment& env) : Substitution(TxnLog::empty(), arena) , builtinTypes(builtinTypes) , arena(arena) , env(env) { } bool isDirty(TypeId ty) override { return env.containsMappedType(ty); } bool isDirty(TypePackId tp) override { return env.containsMappedPack(tp); } TypeId clean(TypeId ty) override { const auto& bounds = env.getMappedTypeBounds(ty); if (bounds.upperBound.empty()) return builtinTypes->unknownType; if (bounds.upperBound.size() == 1) return *begin(bounds.upperBound); return arena->addType(IntersectionType{std::vector(begin(bounds.upperBound), end(bounds.upperBound))}); } TypePackId clean(TypePackId tp) override { if (auto it = env.getMappedPackBounds(tp)) return *it; // Clean is only called when isDirty found a pack bound LUAU_ASSERT(!"Unreachable"); return nullptr; } bool ignoreChildren(TypeId ty) override { if (get(ty)) return true; return ty->persistent; } bool ignoreChildren(TypePackId ty) override { return ty->persistent; } }; std::optional SubtypingEnvironment::applyMappedGenerics(NotNull builtinTypes, NotNull arena, TypeId ty) { ApplyMappedGenerics amg{builtinTypes, arena, *this}; return amg.substitute(ty); } const TypeId* SubtypingEnvironment::tryFindSubstitution(TypeId ty) const { if (auto it = substitutions.find(ty)) return it; if (parent) return parent->tryFindSubstitution(ty); return nullptr; } const SubtypingResult* SubtypingEnvironment::tryFindSubtypingResult(std::pair subAndSuper) const { if (auto it = ephemeralCache.find(subAndSuper)) return it; if (parent) return parent->tryFindSubtypingResult(subAndSuper); return nullptr; } bool SubtypingEnvironment::containsMappedType(TypeId ty) const { if (mappedGenerics.contains(ty)) return true; if (parent) return parent->containsMappedType(ty); return false; } bool SubtypingEnvironment::containsMappedPack(TypePackId tp) const { if (mappedGenericPacks.contains(tp)) return true; if (parent) return parent->containsMappedPack(tp); return false; } SubtypingEnvironment::GenericBounds& SubtypingEnvironment::getMappedTypeBounds(TypeId ty) { if (auto it = mappedGenerics.find(ty)) return *it; if (parent) return parent->getMappedTypeBounds(ty); LUAU_ASSERT(!"Use containsMappedType before asking for bounds!"); return mappedGenerics[ty]; } TypePackId* SubtypingEnvironment::getMappedPackBounds(TypePackId tp) { if (auto it = mappedGenericPacks.find(tp)) return it; if (parent) return parent->getMappedPackBounds(tp); // This fallback is reachable in valid cases, unlike the final part of getMappedTypeBounds return nullptr; } Subtyping::Subtyping( NotNull builtinTypes, NotNull typeArena, NotNull simplifier, NotNull normalizer, NotNull typeFunctionRuntime, NotNull iceReporter ) : builtinTypes(builtinTypes) , arena(typeArena) , simplifier(simplifier) , normalizer(normalizer) , typeFunctionRuntime(typeFunctionRuntime) , iceReporter(iceReporter) { } SubtypingResult Subtyping::isSubtype(TypeId subTy, TypeId superTy, NotNull scope) { SubtypingEnvironment env; SubtypingResult result = isCovariantWith(env, subTy, superTy, scope); for (const auto& [subTy, bounds] : env.mappedGenerics) { const auto& lb = bounds.lowerBound; const auto& ub = bounds.upperBound; TypeId lowerBound = makeAggregateType(lb, builtinTypes->neverType); TypeId upperBound = makeAggregateType(ub, builtinTypes->unknownType); std::shared_ptr nt = normalizer->normalize(upperBound); // we say that the result is true if normalization failed because complex types are likely to be inhabited. NormalizationResult res = nt ? normalizer->isInhabited(nt.get()) : NormalizationResult::True; if (!nt || res == NormalizationResult::HitLimits) result.normalizationTooComplex = true; else if (res == NormalizationResult::False) { /* If the normalized upper bound we're mapping to a generic is * uninhabited, then we must consider the subtyping relation not to * hold. * * This happens eg in () -> (T, T) <: () -> (string, number) * * T appears in covariant position and would have to be both string * and number at once. * * No actual value is both a string and a number, so the test fails. * * TODO: We'll need to add explanitory context here. */ result.isSubtype = false; } SubtypingEnvironment boundsEnv; boundsEnv.parent = &env; SubtypingResult boundsResult = isCovariantWith(boundsEnv, lowerBound, upperBound, scope); boundsResult.reasoning.clear(); result.andAlso(boundsResult); } /* TODO: We presently don't store subtype test results in the persistent * cache if the left-side type is a generic function. * * The implementation would be a bit tricky and we haven't seen any material * impact on benchmarks. * * What we would want to do is to remember points within the type where * mapped generics are introduced. When all the contingent generics are * introduced at which we're doing the test, we can mark the result as * cacheable. */ if (result.isCacheable) resultCache[{subTy, superTy}] = result; return result; } SubtypingResult Subtyping::isSubtype(TypePackId subTp, TypePackId superTp, NotNull scope) { SubtypingEnvironment env; return isCovariantWith(env, subTp, superTp, scope); } SubtypingResult Subtyping::cache(SubtypingEnvironment& env, SubtypingResult result, TypeId subTy, TypeId superTy) { const std::pair p{subTy, superTy}; if (result.isCacheable) resultCache[p] = result; else env.ephemeralCache[p] = result; return result; } 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(SubtypingEnvironment& env, TypeId subTy, TypeId superTy, NotNull scope) { UnifierCounters& counters = normalizer->sharedState->counters; RecursionCounter rc(&counters.recursionCount); if (counters.recursionLimit > 0 && counters.recursionLimit < counters.recursionCount) { SubtypingResult result; result.normalizationTooComplex = true; return result; } subTy = follow(subTy); superTy = follow(superTy); if (const TypeId* subIt = env.tryFindSubstitution(subTy); subIt && *subIt) subTy = *subIt; if (const TypeId* superIt = env.tryFindSubstitution(superTy); superIt && *superIt) superTy = *superIt; const SubtypingResult* cachedResult = resultCache.find({subTy, superTy}); if (cachedResult) return *cachedResult; cachedResult = env.tryFindSubtypingResult({subTy, superTy}); if (cachedResult) return *cachedResult; // 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)) { /* TODO: Caching results for recursive types is really tricky to think * about. * * We'd like to cache at the outermost level where we encounter the * recursive type, but we do not want to cache interior results that * involve the cycle. * * Presently, we stop at cycles and assume that the subtype check will * succeed because we'll eventually get there if it won't. However, if * that cyclic type turns out not to have the asked-for subtyping * relation, then all the intermediate cached results that were * contingent on that assumption need to be evicted from the cache, or * not entered into the cache, or something. * * For now, we do the conservative thing and refuse to cache anything * that touches a cycle. */ SubtypingResult res; res.isSubtype = true; res.isCacheable = false; return res; } 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(env, builtinTypes->neverType, superTy, scope); if (auto superGeneric = get(superTy); superGeneric && subsumes(superGeneric->scope, scope)) return isCovariantWith(env, subTy, builtinTypes->unknownType, scope); SubtypingResult result; if (auto subUnion = get(subTy)) result = isCovariantWith(env, subUnion, superTy, scope); else if (auto superUnion = get(superTy)) { result = isCovariantWith(env, subTy, superUnion, scope); if (!result.isSubtype && !result.normalizationTooComplex) { SubtypingResult semantic = isCovariantWith(env, normalizer->normalize(subTy), normalizer->normalize(superTy), scope); if (semantic.isSubtype) { semantic.reasoning.clear(); result = semantic; } } } else if (auto superIntersection = get(superTy)) result = isCovariantWith(env, subTy, superIntersection, scope); else if (auto subIntersection = get(subTy)) { result = isCovariantWith(env, subIntersection, superTy, scope); if (!result.isSubtype && !result.normalizationTooComplex) { SubtypingResult semantic = isCovariantWith(env, normalizer->normalize(subTy), normalizer->normalize(superTy), scope); if (semantic.isSubtype) { // Clear the semantic reasoning, as any reasonings within // potentially contain invalid paths. semantic.reasoning.clear(); result = semantic; } } } else if (get(superTy)) result = {true}; // We have added this as an exception - the set of inhabitants of any is exactly the set of inhabitants of unknown (since error has no // inhabitants). any = err | unknown, so under semantic subtyping, {} U unknown = unknown else if (get(subTy) && get(superTy)) result = {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 result = isCovariantWith(env, builtinTypes->unknownType, superTy, scope).andAlso(isCovariantWith(env, builtinTypes->errorType, superTy, scope)); } 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); result = {!errorSuppressing}; } else if (get(subTy)) result = {true}; else if (get(superTy)) result = {false}; else if (get(subTy)) result = {true}; else if (auto p = get2(subTy, superTy)) result = isCovariantWith(env, p.first->ty, p.second->ty, scope).withBothComponent(TypePath::TypeField::Negated); else if (auto subNegation = get(subTy)) { result = isCovariantWith(env, subNegation, superTy, scope); if (!result.isSubtype && !result.normalizationTooComplex) { SubtypingResult semantic = isCovariantWith(env, normalizer->normalize(subTy), normalizer->normalize(superTy), scope); if (semantic.isSubtype) { semantic.reasoning.clear(); result = semantic; } } } else if (auto superNegation = get(superTy)) { result = isCovariantWith(env, subTy, superNegation, scope); if (!result.isSubtype && !result.normalizationTooComplex) { SubtypingResult semantic = isCovariantWith(env, normalizer->normalize(subTy), normalizer->normalize(superTy), scope); if (semantic.isSubtype) { semantic.reasoning.clear(); result = semantic; } } } else if (auto subTypeFunctionInstance = get(subTy)) { if (auto substSubTy = env.applyMappedGenerics(builtinTypes, arena, subTy)) subTypeFunctionInstance = get(*substSubTy); result = isCovariantWith(env, subTypeFunctionInstance, superTy, scope); } else if (auto superTypeFunctionInstance = get(superTy)) { if (auto substSuperTy = env.applyMappedGenerics(builtinTypes, arena, superTy)) superTypeFunctionInstance = get(*substSuperTy); result = isCovariantWith(env, subTy, superTypeFunctionInstance, scope); } else if (auto subGeneric = get(subTy); subGeneric && variance == Variance::Covariant) { bool ok = bindGeneric(env, subTy, superTy); result.isSubtype = ok; result.isCacheable = false; } else if (auto superGeneric = get(superTy); superGeneric && variance == Variance::Contravariant) { bool ok = bindGeneric(env, subTy, superTy); result.isSubtype = ok; result.isCacheable = false; } else if (auto p = get2(subTy, superTy)) result = isCovariantWith(env, p, scope); else if (auto p = get2(subTy, superTy)) result = isCovariantWith(env, p, scope); else if (auto p = get2(subTy, superTy)) result = isCovariantWith(env, p, scope); else if (auto p = get2(subTy, superTy)) { auto [subFunction, superPrimitive] = p; result.isSubtype = superPrimitive->type == PrimitiveType::Function; } else if (auto p = get2(subTy, superTy)) result = isCovariantWith(env, p, scope); else if (auto p = get2(subTy, superTy)) result = isCovariantWith(env, p, scope); else if (auto p = get2(subTy, superTy)) result = isCovariantWith(env, p, scope); else if (auto p = get2(subTy, superTy)) result = isCovariantWith(env, p, scope); else if (auto p = get2(subTy, superTy)) result = isCovariantWith(env, p, scope); else if (auto p = get2(subTy, superTy)) result = isCovariantWith(env, subTy, p.first, superTy, p.second, scope); else if (auto p = get2(subTy, superTy)) result = isCovariantWith(env, p, scope); else if (auto p = get2(subTy, superTy)) result = isCovariantWith(env, p, scope); else if (auto p = get2(subTy, superTy)) result = isCovariantWith(env, p, scope); assertReasoningValid(subTy, superTy, result, builtinTypes); return cache(env, result, subTy, superTy); } SubtypingResult Subtyping::isCovariantWith(SubtypingEnvironment& env, TypePackId subTp, TypePackId superTp, NotNull scope) { 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(env, subHead[i], superHead[i], scope).withBothComponent(TypePath::Index{i})); // 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(env, vt->ty, superHead[i], scope) .withSubPath(TypePath::PathBuilder().tail().variadic().build()) .withSuperComponent(TypePath::Index{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 = env.getMappedPackBounds(*subTail)) // TODO: TypePath can't express "slice of a pack + its tail". results.push_back(isCovariantWith(env, *other, superTailPack, scope).withSubComponent(TypePath::PackField::Tail)); else env.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 SubtypingResult{false}.withSubComponent(TypePath::PackField::Tail); } } else if (get(*subTail)) return SubtypingResult{true}.withSubComponent(TypePath::PackField::Tail); else return SubtypingResult{false} .withSubComponent(TypePath::PackField::Tail) .withError({scope->location, UnexpectedTypePackInSubtyping{*subTail}}); } else { results.push_back({false}); return SubtypingResult::all(results); } } 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(env, subHead[i], vt->ty, scope) .withSubComponent(TypePath::Index{i}) .withSuperPath(TypePath::PathBuilder().tail().variadic().build())); } 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 = env.getMappedPackBounds(*superTail)) // TODO: TypePath can't express "slice of a pack + its tail". results.push_back(isContravariantWith(env, subTailPack, *other, scope).withSuperComponent(TypePath::PackField::Tail)); else env.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 SubtypingResult{false}.withSuperComponent(TypePath::PackField::Tail); } } else if (get(*superTail)) return SubtypingResult{true}.withSuperComponent(TypePath::PackField::Tail); else return SubtypingResult{false} .withSuperComponent(TypePath::PackField::Tail) .withError({scope->location, UnexpectedTypePackInSubtyping{FFlag::LuauSubtypingFixTailPack ? *superTail : *subTail}}); } else return {false}; } // Handle tails if (subTail && superTail) { if (auto p = get2(*subTail, *superTail)) { // Variadic component is added by the isCovariantWith // implementation; no need to add it here. results.push_back(isCovariantWith(env, p, scope).withBothComponent(TypePath::PackField::Tail)); } else if (auto p = get2(*subTail, *superTail)) { bool ok = bindGeneric(env, *subTail, *superTail); results.push_back(SubtypingResult{ok}.withBothComponent(TypePath::PackField::Tail)); } else if (auto p = get2(*subTail, *superTail)) { if (variance == Variance::Contravariant) { // (A...) -> number <: (...number) -> number bool ok = bindGeneric(env, *subTail, *superTail); results.push_back(SubtypingResult{ok}.withBothComponent(TypePath::PackField::Tail)); } else { // (number) -> ...number (number) -> A... results.push_back(SubtypingResult{false}.withBothComponent(TypePath::PackField::Tail)); } } else if (auto p = get2(*subTail, *superTail)) { if (TypeId t = follow(p.second->ty); get(t) || get(t)) { // Extra magic rule: // T... <: ...any // T... <: ...unknown // // See https://github.com/luau-lang/luau/issues/767 } else if (variance == Variance::Contravariant) { // (...number) -> number (A...) -> number results.push_back(SubtypingResult{false}.withBothComponent(TypePath::PackField::Tail)); } else { // () -> A... <: () -> ...number bool ok = bindGeneric(env, *subTail, *superTail); results.push_back(SubtypingResult{ok}.withBothComponent(TypePath::PackField::Tail)); } } else if (get(*subTail) || get(*superTail)) // error type is fine on either side results.push_back(SubtypingResult{true}.withBothComponent(TypePath::PackField::Tail)); else return SubtypingResult{false} .withBothComponent(TypePath::PackField::Tail) .withError({scope->location, UnexpectedTypePackInSubtyping{*subTail}}) .withError({scope->location, UnexpectedTypePackInSubtyping{*superTail}}); } else if (subTail) { if (get(*subTail)) { return SubtypingResult{false}.withSubComponent(TypePath::PackField::Tail); } else if (get(*subTail)) { bool ok = bindGeneric(env, *subTail, builtinTypes->emptyTypePack); return SubtypingResult{ok}.withSubComponent(TypePath::PackField::Tail); } else return SubtypingResult{false} .withSubComponent(TypePath::PackField::Tail) .withError({scope->location, UnexpectedTypePackInSubtyping{*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(env, builtinTypes->emptyTypePack, *superTail); results.push_back(SubtypingResult{ok}.withSuperComponent(TypePath::PackField::Tail)); } else results.push_back(SubtypingResult{false}.withSuperComponent(TypePath::PackField::Tail)); } else return SubtypingResult{false} .withSuperComponent(TypePath::PackField::Tail) .withError({scope->location, UnexpectedTypePackInSubtyping{*superTail}}); } SubtypingResult result = SubtypingResult::all(results); assertReasoningValid(subTp, superTp, result, builtinTypes); return result; } template SubtypingResult Subtyping::isContravariantWith(SubtypingEnvironment& env, SubTy&& subTy, SuperTy&& superTy, NotNull scope) { VarianceFlipper vf{&variance}; SubtypingResult result = isCovariantWith(env, superTy, subTy, scope); if (result.reasoning.empty()) result.reasoning.insert(SubtypingReasoning{TypePath::kEmpty, TypePath::kEmpty, SubtypingVariance::Contravariant}); else { // If we don't swap the paths here, we will end up producing an invalid path // whenever we involve contravariance. We'll end up appending path // components that should belong to the supertype to the subtype, and vice // versa. for (auto& reasoning : result.reasoning) { std::swap(reasoning.subPath, reasoning.superPath); // Also swap covariant/contravariant, since those are also the other way // around. if (reasoning.variance == SubtypingVariance::Covariant) reasoning.variance = SubtypingVariance::Contravariant; else if (reasoning.variance == SubtypingVariance::Contravariant) reasoning.variance = SubtypingVariance::Covariant; } } assertReasoningValid(subTy, superTy, result, builtinTypes); return result; } template SubtypingResult Subtyping::isInvariantWith(SubtypingEnvironment& env, SubTy&& subTy, SuperTy&& superTy, NotNull scope) { SubtypingResult result = isCovariantWith(env, subTy, superTy, scope).andAlso(isContravariantWith(env, subTy, superTy, scope)); if (result.reasoning.empty()) result.reasoning.insert(SubtypingReasoning{TypePath::kEmpty, TypePath::kEmpty, SubtypingVariance::Invariant}); else { for (auto& reasoning : result.reasoning) reasoning.variance = SubtypingVariance::Invariant; } assertReasoningValid(subTy, superTy, result, builtinTypes); return result; } template SubtypingResult Subtyping::isCovariantWith(SubtypingEnvironment& env, const TryPair& pair, NotNull scope) { return isCovariantWith(env, pair.first, pair.second, scope); } template SubtypingResult Subtyping::isContravariantWith(SubtypingEnvironment& env, const TryPair& pair, NotNull scope) { return isContravariantWith(env, pair.first, pair.second, scope); } template SubtypingResult Subtyping::isInvariantWith(SubtypingEnvironment& env, const TryPair& pair, NotNull scope) { return isInvariantWith(env, pair.first, pair.second); } /* * 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(SubtypingEnvironment& env, TypeId subTy, const UnionType* superUnion, NotNull scope) { // As per TAPL: T <: A | B iff T <: A || T <: B for (TypeId ty : superUnion) { SubtypingResult next = isCovariantWith(env, subTy, ty, scope); if (next.isSubtype) return SubtypingResult{true}; } /* * TODO: Is it possible here to use the context produced by the above * isCovariantWith() calls to produce a richer, more helpful result in the * case that the subtyping relation does not hold? */ return SubtypingResult{false}; } SubtypingResult Subtyping::isCovariantWith(SubtypingEnvironment& env, const UnionType* subUnion, TypeId superTy, NotNull scope) { // As per TAPL: A | B <: T iff A <: T && B <: T std::vector subtypings; size_t i = 0; for (TypeId ty : subUnion) subtypings.push_back(isCovariantWith(env, ty, superTy, scope).withSubComponent(TypePath::Index{i++})); return SubtypingResult::all(subtypings); } SubtypingResult Subtyping::isCovariantWith(SubtypingEnvironment& env, TypeId subTy, const IntersectionType* superIntersection, NotNull scope) { // As per TAPL: T <: A & B iff T <: A && T <: B std::vector subtypings; size_t i = 0; for (TypeId ty : superIntersection) subtypings.push_back(isCovariantWith(env, subTy, ty, scope).withSuperComponent(TypePath::Index{i++})); return SubtypingResult::all(subtypings); } SubtypingResult Subtyping::isCovariantWith(SubtypingEnvironment& env, const IntersectionType* subIntersection, TypeId superTy, NotNull scope) { // As per TAPL: A & B <: T iff A <: T || B <: T std::vector subtypings; size_t i = 0; for (TypeId ty : subIntersection) subtypings.push_back(isCovariantWith(env, ty, superTy, scope).withSubComponent(TypePath::Index{i++})); return SubtypingResult::any(subtypings); } SubtypingResult Subtyping::isCovariantWith(SubtypingEnvironment& env, const NegationType* subNegation, TypeId superTy, NotNull scope) { TypeId negatedTy = follow(subNegation->ty); SubtypingResult result; // 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 result = isCovariantWith(env, builtinTypes->unknownType, superTy, scope).withSubComponent(TypePath::TypeField::Negated); } else if (is(negatedTy)) { // ¬unknown ~ never result = isCovariantWith(env, builtinTypes->neverType, superTy, scope).withSubComponent(TypePath::TypeField::Negated); } else if (is(negatedTy)) { // ¬any ~ any result = isCovariantWith(env, negatedTy, superTy, scope).withSubComponent(TypePath::TypeField::Negated); } 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(env, negatedPart->ty, superTy, scope).withSubComponent(TypePath::TypeField::Negated)); else { NegationType negatedTmp{ty}; subtypings.push_back(isCovariantWith(env, &negatedTmp, superTy, scope)); } } result = 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(env, negatedPart->ty, superTy, scope).withSubComponent(TypePath::TypeField::Negated)); else { NegationType negatedTmp{ty}; subtypings.push_back(isCovariantWith(env, &negatedTmp, superTy, scope)); } } result = 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 { result = SubtypingResult{false}.withSubComponent(TypePath::TypeField::Negated); } return result; } SubtypingResult Subtyping::isCovariantWith(SubtypingEnvironment& env, const TypeId subTy, const NegationType* superNegation, NotNull scope) { TypeId negatedTy = follow(superNegation->ty); SubtypingResult result; if (is(negatedTy)) { // ¬never ~ unknown result = isCovariantWith(env, subTy, builtinTypes->unknownType, scope); } else if (is(negatedTy)) { // ¬unknown ~ never result = isCovariantWith(env, subTy, builtinTypes->neverType, scope); } else if (is(negatedTy)) { // ¬any ~ any result = isSubtype(subTy, negatedTy, scope); } 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(env, subTy, negatedPart->ty, scope)); else { NegationType negatedTmp{ty}; subtypings.push_back(isCovariantWith(env, subTy, &negatedTmp, scope)); } } 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(env, subTy, negatedPart->ty, scope)); else { NegationType negatedTmp{ty}; subtypings.push_back(isCovariantWith(env, subTy, &negatedTmp, scope)); } } 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) result = {false}; // false (p.first) && p.second->type == PrimitiveType::Boolean) result = {false}; // other cases are true else result = {true}; } else if (auto p = get2(subTy, negatedTy)) { if (p.first->type == PrimitiveType::String && get(p.second)) result = {false}; else if (p.first->type == PrimitiveType::Boolean && get(p.second)) result = {false}; else result = {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)) result = {true}; else if (auto p = get(negatedTy); p && is(subTy)) result = {p->type != PrimitiveType::Table}; else if (auto p = get2(subTy, negatedTy)) result = {p.second->type != PrimitiveType::Function}; else if (auto p = get2(subTy, negatedTy)) result = {*p.first != *p.second}; else if (auto p = get2(subTy, negatedTy)) result = SubtypingResult::negate(isCovariantWith(env, p.first, p.second, scope)); else if (get2(subTy, negatedTy)) result = {true}; else if (is(negatedTy)) iceReporter->ice("attempting to negate a non-testable type"); else result = {false}; return result.withSuperComponent(TypePath::TypeField::Negated); } SubtypingResult Subtyping::isCovariantWith( SubtypingEnvironment& env, const PrimitiveType* subPrim, const PrimitiveType* superPrim, NotNull scope ) { return {subPrim->type == superPrim->type}; } SubtypingResult Subtyping::isCovariantWith( SubtypingEnvironment& env, const SingletonType* subSingleton, const PrimitiveType* superPrim, NotNull scope ) { 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( SubtypingEnvironment& env, const SingletonType* subSingleton, const SingletonType* superSingleton, NotNull scope ) { return {*subSingleton == *superSingleton}; } SubtypingResult Subtyping::isCovariantWith(SubtypingEnvironment& env, const TableType* subTable, const TableType* superTable, NotNull scope) { SubtypingResult result{true}; if (subTable->props.empty() && !subTable->indexer && superTable->indexer) return {false}; for (const auto& [name, superProp] : superTable->props) { std::vector results; if (auto subIter = subTable->props.find(name); subIter != subTable->props.end()) results.push_back(isCovariantWith(env, subIter->second, superProp, name, scope)); else if (subTable->indexer) { if (isCovariantWith(env, builtinTypes->stringType, subTable->indexer->indexType, scope).isSubtype) { if (superProp.isShared()) results.push_back(isInvariantWith(env, subTable->indexer->indexResultType, superProp.type(), scope) .withSubComponent(TypePath::TypeField::IndexResult) .withSuperComponent(TypePath::Property::read(name))); else { if (superProp.readTy) results.push_back(isCovariantWith(env, subTable->indexer->indexResultType, *superProp.readTy, scope) .withSubComponent(TypePath::TypeField::IndexResult) .withSuperComponent(TypePath::Property::read(name))); if (superProp.writeTy) results.push_back(isContravariantWith(env, subTable->indexer->indexResultType, *superProp.writeTy, scope) .withSubComponent(TypePath::TypeField::IndexResult) .withSuperComponent(TypePath::Property::write(name))); } } } if (results.empty()) return SubtypingResult{false}; result.andAlso(SubtypingResult::all(results)); } if (superTable->indexer) { if (subTable->indexer) result.andAlso(isInvariantWith(env, *subTable->indexer, *superTable->indexer, scope)); else return {false}; } return result; } SubtypingResult Subtyping::isCovariantWith(SubtypingEnvironment& env, const MetatableType* subMt, const MetatableType* superMt, NotNull scope) { return isCovariantWith(env, subMt->table, superMt->table, scope) .withBothComponent(TypePath::TypeField::Table) .andAlso(isCovariantWith(env, subMt->metatable, superMt->metatable, scope).withBothComponent(TypePath::TypeField::Metatable)); } SubtypingResult Subtyping::isCovariantWith(SubtypingEnvironment& env, const MetatableType* subMt, const TableType* superTable, NotNull scope) { if (auto subTable = get(follow(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(env, subTable, superTable, scope); } else { // TODO: This may be a case we actually hit? return {false}; } } SubtypingResult Subtyping::isCovariantWith(SubtypingEnvironment& env, const ClassType* subClass, const ClassType* superClass, NotNull scope) { return {isSubclass(subClass, superClass)}; } SubtypingResult Subtyping::isCovariantWith( SubtypingEnvironment& env, TypeId subTy, const ClassType* subClass, TypeId superTy, const TableType* superTable, NotNull scope ) { SubtypingResult result{true}; env.substitutions[superTy] = subTy; for (const auto& [name, prop] : superTable->props) { if (auto classProp = lookupClassProp(subClass, name)) { result.andAlso(isCovariantWith(env, *classProp, prop, name, scope)); } else { result = {false}; break; } } env.substitutions[superTy] = nullptr; return result; } SubtypingResult Subtyping::isCovariantWith( SubtypingEnvironment& env, const FunctionType* subFunction, const FunctionType* superFunction, NotNull scope ) { SubtypingResult result; { result.orElse( isContravariantWith(env, subFunction->argTypes, superFunction->argTypes, scope).withBothComponent(TypePath::PackField::Arguments) ); // If subtyping failed in the argument packs, we should check if there's a hidden variadic tail and try ignoring it. // This might cause subtyping correctly because the sub type here may not have a hidden variadic tail or equivalent. if (!result.isSubtype) { auto [arguments, tail] = flatten(superFunction->argTypes); if (auto variadic = get(tail); variadic && variadic->hidden) { result.orElse(isContravariantWith(env, subFunction->argTypes, arena->addTypePack(TypePack{arguments}), scope) .withBothComponent(TypePath::PackField::Arguments)); } } } result.andAlso(isCovariantWith(env, subFunction->retTypes, superFunction->retTypes, scope).withBothComponent(TypePath::PackField::Returns)); return result; } SubtypingResult Subtyping::isCovariantWith(SubtypingEnvironment& env, const TableType* subTable, const PrimitiveType* superPrim, NotNull scope) { SubtypingResult result{false}; if (superPrim->type == PrimitiveType::Table) result.isSubtype = true; return result; } SubtypingResult Subtyping::isCovariantWith(SubtypingEnvironment& env, const PrimitiveType* subPrim, const TableType* superTable, NotNull scope) { 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(env, stringTable, superTable, scope).withSubPath(TypePath::PathBuilder().mt().readProp("__index").build()) ); } } } } else if (subPrim->type == PrimitiveType::Table) { const bool isSubtype = superTable->props.empty() && !superTable->indexer.has_value(); return {isSubtype}; } return result; } SubtypingResult Subtyping::isCovariantWith( SubtypingEnvironment& env, const SingletonType* subSingleton, const TableType* superTable, NotNull scope ) { 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(env, stringTable, superTable, scope).withSubPath(TypePath::PathBuilder().mt().readProp("__index").build()) ); } } } } return result; } SubtypingResult Subtyping::isCovariantWith( SubtypingEnvironment& env, const TableIndexer& subIndexer, const TableIndexer& superIndexer, NotNull scope ) { return isInvariantWith(env, subIndexer.indexType, superIndexer.indexType, scope) .withBothComponent(TypePath::TypeField::IndexLookup) .andAlso( isInvariantWith(env, subIndexer.indexResultType, superIndexer.indexResultType, scope).withBothComponent(TypePath::TypeField::IndexResult) ); } SubtypingResult Subtyping::isCovariantWith( SubtypingEnvironment& env, const Property& subProp, const Property& superProp, const std::string& name, NotNull scope ) { SubtypingResult res{true}; if (superProp.isShared() && subProp.isShared()) res.andAlso(isInvariantWith(env, subProp.type(), superProp.type(), scope).withBothComponent(TypePath::Property::read(name))); else { if (superProp.readTy.has_value() && subProp.readTy.has_value()) res.andAlso(isCovariantWith(env, *subProp.readTy, *superProp.readTy, scope).withBothComponent(TypePath::Property::read(name))); if (superProp.writeTy.has_value() && subProp.writeTy.has_value()) res.andAlso(isContravariantWith(env, *subProp.writeTy, *superProp.writeTy, scope).withBothComponent(TypePath::Property::write(name))); if (superProp.isReadWrite()) { if (subProp.isReadOnly()) res.andAlso(SubtypingResult{false}.withBothComponent(TypePath::Property::read(name))); else if (subProp.isWriteOnly()) res.andAlso(SubtypingResult{false}.withBothComponent(TypePath::Property::write(name))); } } return res; } SubtypingResult Subtyping::isCovariantWith( SubtypingEnvironment& env, const std::shared_ptr& subNorm, const std::shared_ptr& superNorm, NotNull scope ) { if (!subNorm || !superNorm) return {false, true}; SubtypingResult result = isCovariantWith(env, subNorm->tops, superNorm->tops, scope); result.andAlso(isCovariantWith(env, subNorm->booleans, superNorm->booleans, scope)); result.andAlso( isCovariantWith(env, subNorm->classes, superNorm->classes, scope).orElse(isCovariantWith(env, subNorm->classes, superNorm->tables, scope)) ); result.andAlso(isCovariantWith(env, subNorm->errors, superNorm->errors, scope)); result.andAlso(isCovariantWith(env, subNorm->nils, superNorm->nils, scope)); result.andAlso(isCovariantWith(env, subNorm->numbers, superNorm->numbers, scope)); result.andAlso(isCovariantWith(env, subNorm->strings, superNorm->strings, scope)); result.andAlso(isCovariantWith(env, subNorm->strings, superNorm->tables, scope)); result.andAlso(isCovariantWith(env, subNorm->threads, superNorm->threads, scope)); result.andAlso(isCovariantWith(env, subNorm->buffers, superNorm->buffers, scope)); result.andAlso(isCovariantWith(env, subNorm->tables, superNorm->tables, scope)); result.andAlso(isCovariantWith(env, subNorm->functions, superNorm->functions, scope)); // isCovariantWith(subNorm->tyvars, superNorm->tyvars); return result; } SubtypingResult Subtyping::isCovariantWith( SubtypingEnvironment& env, const NormalizedClassType& subClass, const NormalizedClassType& superClass, NotNull scope ) { for (const auto& [subClassTy, _] : subClass.classes) { SubtypingResult result; for (const auto& [superClassTy, superNegations] : superClass.classes) { result.orElse(isCovariantWith(env, subClassTy, superClassTy, scope)); if (!result.isSubtype) continue; for (TypeId negation : superNegations) { result.andAlso(SubtypingResult::negate(isCovariantWith(env, subClassTy, negation, scope))); if (result.isSubtype) break; } } if (!result.isSubtype) return result; } return {true}; } SubtypingResult Subtyping::isCovariantWith( SubtypingEnvironment& env, const NormalizedClassType& subClass, const TypeIds& superTables, NotNull scope ) { for (const auto& [subClassTy, _] : subClass.classes) { SubtypingResult result; for (TypeId superTableTy : superTables) result.orElse(isCovariantWith(env, subClassTy, superTableTy, scope)); if (!result.isSubtype) return result; } return {true}; } SubtypingResult Subtyping::isCovariantWith( SubtypingEnvironment& env, const NormalizedStringType& subString, const NormalizedStringType& superString, NotNull scope ) { bool isSubtype = Luau::isSubtype(subString, superString); return {isSubtype}; } SubtypingResult Subtyping::isCovariantWith( SubtypingEnvironment& env, const NormalizedStringType& subString, const TypeIds& superTables, NotNull scope ) { if (subString.isNever()) return {true}; if (subString.isCofinite) { SubtypingResult result; for (const auto& superTable : superTables) { result.orElse(isCovariantWith(env, builtinTypes->stringType, superTable, scope)); 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(env, subString, superTable, scope)); if (!result.isSubtype) break; } if (!result.isSubtype) continue; else return result; } return {false}; } SubtypingResult Subtyping::isCovariantWith( SubtypingEnvironment& env, const NormalizedFunctionType& subFunction, const NormalizedFunctionType& superFunction, NotNull scope ) { if (subFunction.isNever()) return {true}; else if (superFunction.isTop) return {true}; else return isCovariantWith(env, subFunction.parts, superFunction.parts, scope); } SubtypingResult Subtyping::isCovariantWith(SubtypingEnvironment& env, const TypeIds& subTypes, const TypeIds& superTypes, NotNull scope) { std::vector results; for (TypeId subTy : subTypes) { results.emplace_back(); for (TypeId superTy : superTypes) results.back().orElse(isCovariantWith(env, subTy, superTy, scope)); } return SubtypingResult::all(results); } SubtypingResult Subtyping::isCovariantWith( SubtypingEnvironment& env, const VariadicTypePack* subVariadic, const VariadicTypePack* superVariadic, NotNull scope ) { return isCovariantWith(env, subVariadic->ty, superVariadic->ty, scope).withBothComponent(TypePath::TypeField::Variadic); } bool Subtyping::bindGeneric(SubtypingEnvironment& env, TypeId subTy, TypeId superTy) { if (variance == Variance::Covariant) { if (!get(subTy)) return false; if (!env.mappedGenerics.find(subTy) && env.containsMappedType(subTy)) iceReporter->ice("attempting to modify bounds of a potentially visited generic"); env.mappedGenerics[subTy].upperBound.insert(superTy); } else { if (!get(superTy)) return false; if (!env.mappedGenerics.find(superTy) && env.containsMappedType(superTy)) iceReporter->ice("attempting to modify bounds of a potentially visited generic"); env.mappedGenerics[superTy].lowerBound.insert(subTy); } return true; } SubtypingResult Subtyping::isCovariantWith( SubtypingEnvironment& env, const TypeFunctionInstanceType* subFunctionInstance, const TypeId superTy, NotNull scope ) { // Reduce the type function instance auto [ty, errors] = handleTypeFunctionReductionResult(subFunctionInstance, scope); // If we return optional, that means the type function was irreducible - we can reduce that to never return isCovariantWith(env, ty, superTy, scope).withErrors(errors).withSubComponent(TypePath::Reduction{ty}); } SubtypingResult Subtyping::isCovariantWith( SubtypingEnvironment& env, const TypeId subTy, const TypeFunctionInstanceType* superFunctionInstance, NotNull scope ) { // Reduce the type function instance auto [ty, errors] = handleTypeFunctionReductionResult(superFunctionInstance, scope); return isCovariantWith(env, subTy, ty, scope).withErrors(errors).withSuperComponent(TypePath::Reduction{ty}); } /* * 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(SubtypingEnvironment& env, TypePackId subTp, TypePackId superTp) { if (variance == Variance::Contravariant) std::swap(superTp, subTp); if (!get(subTp)) return false; if (TypePackId* m = env.getMappedPackBounds(subTp)) return *m == superTp; env.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))}); } std::pair Subtyping::handleTypeFunctionReductionResult(const TypeFunctionInstanceType* functionInstance, NotNull scope) { TypeFunctionContext context{arena, builtinTypes, scope, simplifier, normalizer, typeFunctionRuntime, iceReporter, NotNull{&limits}}; TypeId function = arena->addType(*functionInstance); FunctionGraphReductionResult result = reduceTypeFunctions(function, {}, context, true); ErrorVec errors; if (result.blockedTypes.size() != 0 || result.blockedPacks.size() != 0) { errors.push_back(TypeError{{}, UninhabitedTypeFunction{function}}); return {builtinTypes->neverType, errors}; } if (result.reducedTypes.contains(function)) return {function, errors}; return {builtinTypes->neverType, errors}; } } // namespace Luau