// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details #include "Luau/TypeVar.h" #include "Luau/BuiltinDefinitions.h" #include "Luau/Common.h" #include "Luau/DenseHash.h" #include "Luau/Error.h" #include "Luau/RecursionCounter.h" #include "Luau/StringUtils.h" #include "Luau/ToString.h" #include "Luau/TypeInfer.h" #include "Luau/TypePack.h" #include "Luau/VisitTypeVar.h" #include #include #include #include #include LUAU_FASTFLAG(DebugLuauFreezeArena) LUAU_FASTINTVARIABLE(LuauTypeMaximumStringifierLength, 500) LUAU_FASTINTVARIABLE(LuauTableTypeMaximumStringifierLength, 0) LUAU_FASTINT(LuauTypeInferRecursionLimit) LUAU_FASTFLAG(LuauUnknownAndNeverType) LUAU_FASTFLAGVARIABLE(LuauMaybeGenericIntersectionTypes, false) LUAU_FASTFLAGVARIABLE(LuauStringFormatArgumentErrorFix, false) LUAU_FASTFLAGVARIABLE(LuauNoMoreGlobalSingletonTypes, false) LUAU_FASTFLAG(LuauInstantiateInSubtyping) namespace Luau { std::optional> magicFunctionFormat( TypeChecker& typechecker, const ScopePtr& scope, const AstExprCall& expr, WithPredicate withPredicate); static std::optional> magicFunctionGmatch( TypeChecker& typechecker, const ScopePtr& scope, const AstExprCall& expr, WithPredicate withPredicate); static std::optional> magicFunctionMatch( TypeChecker& typechecker, const ScopePtr& scope, const AstExprCall& expr, WithPredicate withPredicate); static std::optional> magicFunctionFind( TypeChecker& typechecker, const ScopePtr& scope, const AstExprCall& expr, WithPredicate withPredicate); TypeId follow(TypeId t) { return follow(t, [](TypeId t) { return t; }); } TypeId follow(TypeId t, std::function mapper) { auto advance = [&mapper](TypeId ty) -> std::optional { if (auto btv = get>(mapper(ty))) return btv->boundTo; else if (auto ttv = get(mapper(ty))) return ttv->boundTo; else return std::nullopt; }; auto force = [&mapper](TypeId ty) { if (auto ltv = get_if(&mapper(ty)->ty)) { TypeId res = ltv->thunk(); if (get(res)) throw std::runtime_error("Lazy TypeVar cannot resolve to another Lazy TypeVar"); *asMutable(ty) = BoundTypeVar(res); } }; force(t); TypeId cycleTester = t; // Null once we've determined that there is no cycle if (auto a = advance(cycleTester)) cycleTester = *a; else return t; while (true) { force(t); auto a1 = advance(t); if (a1) t = *a1; else return t; if (nullptr != cycleTester) { auto a2 = advance(cycleTester); if (a2) { auto a3 = advance(*a2); if (a3) cycleTester = *a3; else cycleTester = nullptr; } else cycleTester = nullptr; if (t == cycleTester) throw std::runtime_error("Luau::follow detected a TypeVar cycle!!"); } } } std::vector flattenIntersection(TypeId ty) { if (!get(follow(ty))) return {ty}; std::unordered_set seen; std::deque queue{ty}; std::vector result; while (!queue.empty()) { TypeId current = follow(queue.front()); queue.pop_front(); if (seen.find(current) != seen.end()) continue; seen.insert(current); if (auto itv = get(current)) { for (TypeId ty : itv->parts) queue.push_back(ty); } else result.push_back(current); } return result; } bool isPrim(TypeId ty, PrimitiveTypeVar::Type primType) { auto p = get(follow(ty)); return p && p->type == primType; } bool isNil(TypeId ty) { return isPrim(ty, PrimitiveTypeVar::NilType); } bool isBoolean(TypeId ty) { if (isPrim(ty, PrimitiveTypeVar::Boolean) || get(get(follow(ty)))) return true; if (auto utv = get(follow(ty))) return std::all_of(begin(utv), end(utv), isBoolean); return false; } bool isNumber(TypeId ty) { return isPrim(ty, PrimitiveTypeVar::Number); } // Returns true when ty is a subtype of string bool isString(TypeId ty) { ty = follow(ty); if (isPrim(ty, PrimitiveTypeVar::String) || get(get(ty))) return true; if (auto utv = get(ty)) return std::all_of(begin(utv), end(utv), isString); return false; } // Returns true when ty is a supertype of string bool maybeString(TypeId ty) { ty = follow(ty); if (isPrim(ty, PrimitiveTypeVar::String) || get(ty)) return true; if (auto utv = get(ty)) return std::any_of(begin(utv), end(utv), maybeString); return false; } bool isThread(TypeId ty) { return isPrim(ty, PrimitiveTypeVar::Thread); } bool isOptional(TypeId ty) { if (isNil(ty)) return true; ty = follow(ty); if (get(ty) || (FFlag::LuauUnknownAndNeverType && get(ty))) return true; auto utv = get(ty); if (!utv) return false; return std::any_of(begin(utv), end(utv), isOptional); } bool isTableIntersection(TypeId ty) { if (!get(follow(ty))) return false; std::vector parts = flattenIntersection(ty); return std::all_of(parts.begin(), parts.end(), getTableType); } bool isOverloadedFunction(TypeId ty) { if (!get(follow(ty))) return false; auto isFunction = [](TypeId part) -> bool { return get(part); }; std::vector parts = flattenIntersection(ty); return std::all_of(parts.begin(), parts.end(), isFunction); } std::optional getMetatable(TypeId type, NotNull singletonTypes) { type = follow(type); if (const MetatableTypeVar* mtType = get(type)) return mtType->metatable; else if (const ClassTypeVar* classType = get(type)) return classType->metatable; else if (isString(type)) { auto ptv = get(singletonTypes->stringType); LUAU_ASSERT(ptv && ptv->metatable); return ptv->metatable; } return std::nullopt; } const TableTypeVar* getTableType(TypeId type) { type = follow(type); if (const TableTypeVar* ttv = get(type)) return ttv; else if (const MetatableTypeVar* mtv = get(type)) return get(follow(mtv->table)); else return nullptr; } TableTypeVar* getMutableTableType(TypeId type) { return const_cast(getTableType(type)); } const std::string* getName(TypeId type) { type = follow(type); if (auto mtv = get(type)) { if (mtv->syntheticName) return &*mtv->syntheticName; type = follow(mtv->table); } if (auto ttv = get(type)) { if (ttv->name) return &*ttv->name; if (ttv->syntheticName) return &*ttv->syntheticName; } return nullptr; } std::optional getDefinitionModuleName(TypeId type) { type = follow(type); if (auto ttv = get(type)) { if (!ttv->definitionModuleName.empty()) return ttv->definitionModuleName; } else if (auto ftv = get(type)) { if (ftv->definition) return ftv->definition->definitionModuleName; } else if (auto ctv = get(type)) { if (!ctv->definitionModuleName.empty()) return ctv->definitionModuleName; } return std::nullopt; } bool isSubset(const UnionTypeVar& super, const UnionTypeVar& sub) { std::unordered_set superTypes; for (TypeId id : super.options) superTypes.insert(id); for (TypeId id : sub.options) { if (superTypes.find(id) == superTypes.end()) return false; } return true; } // When typechecking an assignment `x = e`, we typecheck `x:T` and `e:U`, // then instantiate U if `isGeneric(U)` is true, and `maybeGeneric(T)` is false. bool isGeneric(TypeId ty) { LUAU_ASSERT(!FFlag::LuauInstantiateInSubtyping); ty = follow(ty); if (auto ftv = get(ty)) return ftv->generics.size() > 0 || ftv->genericPacks.size() > 0; else // TODO: recurse on type synonyms CLI-39914 // TODO: recurse on table types CLI-39914 return false; } bool maybeGeneric(TypeId ty) { LUAU_ASSERT(!FFlag::LuauInstantiateInSubtyping); if (FFlag::LuauMaybeGenericIntersectionTypes) { ty = follow(ty); if (get(ty)) return true; if (auto ttv = get(ty)) { // TODO: recurse on table types CLI-39914 (void)ttv; return true; } if (auto itv = get(ty)) { return std::any_of(begin(itv), end(itv), maybeGeneric); } return isGeneric(ty); } ty = follow(ty); if (get(ty)) return true; else if (auto ttv = get(ty)) { // TODO: recurse on table types CLI-39914 (void)ttv; return true; } else return isGeneric(ty); } bool maybeSingleton(TypeId ty) { ty = follow(ty); if (get(ty)) return true; if (const UnionTypeVar* utv = get(ty)) for (TypeId option : utv) if (get(follow(option))) return true; return false; } bool hasLength(TypeId ty, DenseHashSet& seen, int* recursionCount) { RecursionLimiter _rl(recursionCount, FInt::LuauTypeInferRecursionLimit); ty = follow(ty); if (seen.contains(ty)) return true; if (isString(ty) || get(ty) || get(ty) || get(ty)) return true; if (auto uty = get(ty)) { seen.insert(ty); for (TypeId part : uty->options) { if (!hasLength(part, seen, recursionCount)) return false; } return true; } if (auto ity = get(ty)) { seen.insert(ty); for (TypeId part : ity->parts) { if (hasLength(part, seen, recursionCount)) return true; } return false; } return false; } BlockedTypeVar::BlockedTypeVar() : index(++nextIndex) { } int BlockedTypeVar::nextIndex = 0; PendingExpansionTypeVar::PendingExpansionTypeVar( std::optional prefix, AstName name, std::vector typeArguments, std::vector packArguments) : prefix(prefix) , name(name) , typeArguments(typeArguments) , packArguments(packArguments) , index(++nextIndex) { } size_t PendingExpansionTypeVar::nextIndex = 0; FunctionTypeVar::FunctionTypeVar(TypePackId argTypes, TypePackId retTypes, std::optional defn, bool hasSelf) : argTypes(argTypes) , retTypes(retTypes) , definition(std::move(defn)) , hasSelf(hasSelf) { } FunctionTypeVar::FunctionTypeVar(TypeLevel level, TypePackId argTypes, TypePackId retTypes, std::optional defn, bool hasSelf) : level(level) , argTypes(argTypes) , retTypes(retTypes) , definition(std::move(defn)) , hasSelf(hasSelf) { } FunctionTypeVar::FunctionTypeVar( TypeLevel level, Scope* scope, TypePackId argTypes, TypePackId retTypes, std::optional defn, bool hasSelf) : level(level) , scope(scope) , argTypes(argTypes) , retTypes(retTypes) , definition(std::move(defn)) , hasSelf(hasSelf) { } FunctionTypeVar::FunctionTypeVar(std::vector generics, std::vector genericPacks, TypePackId argTypes, TypePackId retTypes, std::optional defn, bool hasSelf) : generics(generics) , genericPacks(genericPacks) , argTypes(argTypes) , retTypes(retTypes) , definition(std::move(defn)) , hasSelf(hasSelf) { } FunctionTypeVar::FunctionTypeVar(TypeLevel level, std::vector generics, std::vector genericPacks, TypePackId argTypes, TypePackId retTypes, std::optional defn, bool hasSelf) : level(level) , generics(generics) , genericPacks(genericPacks) , argTypes(argTypes) , retTypes(retTypes) , definition(std::move(defn)) , hasSelf(hasSelf) { } FunctionTypeVar::FunctionTypeVar(TypeLevel level, Scope* scope, std::vector generics, std::vector genericPacks, TypePackId argTypes, TypePackId retTypes, std::optional defn, bool hasSelf) : level(level) , scope(scope) , generics(generics) , genericPacks(genericPacks) , argTypes(argTypes) , retTypes(retTypes) , definition(std::move(defn)) , hasSelf(hasSelf) { } TableTypeVar::TableTypeVar(TableState state, TypeLevel level, Scope* scope) : state(state) , level(level) , scope(scope) { } TableTypeVar::TableTypeVar(const Props& props, const std::optional& indexer, TypeLevel level, TableState state) : props(props) , indexer(indexer) , state(state) , level(level) { } TableTypeVar::TableTypeVar(const Props& props, const std::optional& indexer, TypeLevel level, Scope* scope, TableState state) : props(props) , indexer(indexer) , state(state) , level(level) , scope(scope) { } // Test TypeVars for equivalence // More complex than we'd like because TypeVars can self-reference. bool areSeen(SeenSet& seen, const void* lhs, const void* rhs) { if (lhs == rhs) return true; auto p = std::make_pair(const_cast(lhs), const_cast(rhs)); if (seen.find(p) != seen.end()) return true; seen.insert(p); return false; } bool areEqual(SeenSet& seen, const FunctionTypeVar& lhs, const FunctionTypeVar& rhs) { if (areSeen(seen, &lhs, &rhs)) return true; // TODO: check generics CLI-39915 if (!areEqual(seen, *lhs.argTypes, *rhs.argTypes)) return false; if (!areEqual(seen, *lhs.retTypes, *rhs.retTypes)) return false; return true; } bool areEqual(SeenSet& seen, const TableTypeVar& lhs, const TableTypeVar& rhs) { if (areSeen(seen, &lhs, &rhs)) return true; if (lhs.state != rhs.state) return false; if (lhs.props.size() != rhs.props.size()) return false; if (bool(lhs.indexer) != bool(rhs.indexer)) return false; if (lhs.indexer && rhs.indexer) { if (!areEqual(seen, *lhs.indexer->indexType, *rhs.indexer->indexType)) return false; if (!areEqual(seen, *lhs.indexer->indexResultType, *rhs.indexer->indexResultType)) return false; } auto l = lhs.props.begin(); auto r = rhs.props.begin(); while (l != lhs.props.end()) { if (l->first != r->first) return false; if (!areEqual(seen, *l->second.type, *r->second.type)) return false; ++l; ++r; } return true; } static bool areEqual(SeenSet& seen, const MetatableTypeVar& lhs, const MetatableTypeVar& rhs) { if (areSeen(seen, &lhs, &rhs)) return true; return areEqual(seen, *lhs.table, *rhs.table) && areEqual(seen, *lhs.metatable, *rhs.metatable); } bool areEqual(SeenSet& seen, const TypeVar& lhs, const TypeVar& rhs) { if (auto bound = get_if(&lhs.ty)) return areEqual(seen, *bound->boundTo, rhs); if (auto bound = get_if(&rhs.ty)) return areEqual(seen, lhs, *bound->boundTo); if (lhs.ty.index() != rhs.ty.index()) return false; { const FreeTypeVar* lf = get_if(&lhs.ty); const FreeTypeVar* rf = get_if(&rhs.ty); if (lf && rf) return lf->index == rf->index; } { const GenericTypeVar* lg = get_if(&lhs.ty); const GenericTypeVar* rg = get_if(&rhs.ty); if (lg && rg) return lg->index == rg->index; } { const PrimitiveTypeVar* lp = get_if(&lhs.ty); const PrimitiveTypeVar* rp = get_if(&rhs.ty); if (lp && rp) return lp->type == rp->type; } { const GenericTypeVar* lg = get_if(&lhs.ty); const GenericTypeVar* rg = get_if(&rhs.ty); if (lg && rg) return lg->index == rg->index; } { const ErrorTypeVar* le = get_if(&lhs.ty); const ErrorTypeVar* re = get_if(&rhs.ty); if (le && re) return le->index == re->index; } { const FunctionTypeVar* lf = get_if(&lhs.ty); const FunctionTypeVar* rf = get_if(&rhs.ty); if (lf && rf) return areEqual(seen, *lf, *rf); } { const TableTypeVar* lt = get_if(&lhs.ty); const TableTypeVar* rt = get_if(&rhs.ty); if (lt && rt) return areEqual(seen, *lt, *rt); } { const MetatableTypeVar* lmt = get_if(&lhs.ty); const MetatableTypeVar* rmt = get_if(&rhs.ty); if (lmt && rmt) return areEqual(seen, *lmt, *rmt); } if (get_if(&lhs.ty) && get_if(&rhs.ty)) return true; return false; } TypeVar* asMutable(TypeId ty) { return const_cast(ty); } bool TypeVar::operator==(const TypeVar& rhs) const { SeenSet seen; return areEqual(seen, *this, rhs); } bool TypeVar::operator!=(const TypeVar& rhs) const { SeenSet seen; return !areEqual(seen, *this, rhs); } TypeVar& TypeVar::operator=(const TypeVariant& rhs) { ty = rhs; return *this; } TypeVar& TypeVar::operator=(TypeVariant&& rhs) { ty = std::move(rhs); return *this; } TypeVar& TypeVar::operator=(const TypeVar& rhs) { LUAU_ASSERT(owningArena == rhs.owningArena); LUAU_ASSERT(!rhs.persistent); reassign(rhs); return *this; } TypeId makeFunction(TypeArena& arena, std::optional selfType, std::initializer_list generics, std::initializer_list genericPacks, std::initializer_list paramTypes, std::initializer_list paramNames, std::initializer_list retTypes); SingletonTypes::SingletonTypes() : arena(new TypeArena) , debugFreezeArena(FFlag::DebugLuauFreezeArena) , nilType(arena->addType(TypeVar{PrimitiveTypeVar{PrimitiveTypeVar::NilType}, /*persistent*/ true})) , numberType(arena->addType(TypeVar{PrimitiveTypeVar{PrimitiveTypeVar::Number}, /*persistent*/ true})) , stringType(arena->addType(TypeVar{PrimitiveTypeVar{PrimitiveTypeVar::String}, /*persistent*/ true})) , booleanType(arena->addType(TypeVar{PrimitiveTypeVar{PrimitiveTypeVar::Boolean}, /*persistent*/ true})) , threadType(arena->addType(TypeVar{PrimitiveTypeVar{PrimitiveTypeVar::Thread}, /*persistent*/ true})) , trueType(arena->addType(TypeVar{SingletonTypeVar{BooleanSingleton{true}}, /*persistent*/ true})) , falseType(arena->addType(TypeVar{SingletonTypeVar{BooleanSingleton{false}}, /*persistent*/ true})) , anyType(arena->addType(TypeVar{AnyTypeVar{}, /*persistent*/ true})) , unknownType(arena->addType(TypeVar{UnknownTypeVar{}, /*persistent*/ true})) , neverType(arena->addType(TypeVar{NeverTypeVar{}, /*persistent*/ true})) , errorType(arena->addType(TypeVar{ErrorTypeVar{}, /*persistent*/ true})) , anyTypePack(arena->addTypePack(TypePackVar{VariadicTypePack{anyType}, /*persistent*/ true})) , neverTypePack(arena->addTypePack(TypePackVar{VariadicTypePack{neverType}, /*persistent*/ true})) , uninhabitableTypePack(arena->addTypePack({neverType}, neverTypePack)) , errorTypePack(arena->addTypePack(TypePackVar{Unifiable::Error{}, /*persistent*/ true})) { TypeId stringMetatable = makeStringMetatable(); asMutable(stringType)->ty = PrimitiveTypeVar{PrimitiveTypeVar::String, stringMetatable}; persist(stringMetatable); persist(uninhabitableTypePack); freeze(*arena); } SingletonTypes::~SingletonTypes() { // Destroy the arena with the same memory management flags it was created with bool prevFlag = FFlag::DebugLuauFreezeArena; FFlag::DebugLuauFreezeArena.value = debugFreezeArena; unfreeze(*arena); arena.reset(nullptr); FFlag::DebugLuauFreezeArena.value = prevFlag; } TypeId SingletonTypes::makeStringMetatable() { const TypeId optionalNumber = arena->addType(UnionTypeVar{{nilType, numberType}}); const TypeId optionalString = arena->addType(UnionTypeVar{{nilType, stringType}}); const TypeId optionalBoolean = arena->addType(UnionTypeVar{{nilType, booleanType}}); const TypePackId oneStringPack = arena->addTypePack({stringType}); const TypePackId anyTypePack = arena->addTypePack(TypePackVar{VariadicTypePack{anyType}, true}); FunctionTypeVar formatFTV{arena->addTypePack(TypePack{{stringType}, anyTypePack}), oneStringPack}; formatFTV.magicFunction = &magicFunctionFormat; const TypeId formatFn = arena->addType(formatFTV); const TypePackId emptyPack = arena->addTypePack({}); const TypePackId stringVariadicList = arena->addTypePack(TypePackVar{VariadicTypePack{stringType}}); const TypePackId numberVariadicList = arena->addTypePack(TypePackVar{VariadicTypePack{numberType}}); const TypeId stringToStringType = makeFunction(*arena, std::nullopt, {}, {}, {stringType}, {}, {stringType}); const TypeId replArgType = arena->addType( UnionTypeVar{{stringType, arena->addType(TableTypeVar({}, TableIndexer(stringType, stringType), TypeLevel{}, TableState::Generic)), makeFunction(*arena, std::nullopt, {}, {}, {stringType}, {}, {stringType})}}); const TypeId gsubFunc = makeFunction(*arena, stringType, {}, {}, {stringType, replArgType, optionalNumber}, {}, {stringType, numberType}); const TypeId gmatchFunc = makeFunction(*arena, stringType, {}, {}, {stringType}, {}, {arena->addType(FunctionTypeVar{emptyPack, stringVariadicList})}); attachMagicFunction(gmatchFunc, magicFunctionGmatch); const TypeId matchFunc = arena->addType( FunctionTypeVar{arena->addTypePack({stringType, stringType, optionalNumber}), arena->addTypePack(TypePackVar{VariadicTypePack{stringType}})}); attachMagicFunction(matchFunc, magicFunctionMatch); const TypeId findFunc = arena->addType(FunctionTypeVar{arena->addTypePack({stringType, stringType, optionalNumber, optionalBoolean}), arena->addTypePack(TypePack{{optionalNumber, optionalNumber}, stringVariadicList})}); attachMagicFunction(findFunc, magicFunctionFind); TableTypeVar::Props stringLib = { {"byte", {arena->addType(FunctionTypeVar{arena->addTypePack({stringType, optionalNumber, optionalNumber}), numberVariadicList})}}, {"char", {arena->addType(FunctionTypeVar{numberVariadicList, arena->addTypePack({stringType})})}}, {"find", {findFunc}}, {"format", {formatFn}}, // FIXME {"gmatch", {gmatchFunc}}, {"gsub", {gsubFunc}}, {"len", {makeFunction(*arena, stringType, {}, {}, {}, {}, {numberType})}}, {"lower", {stringToStringType}}, {"match", {matchFunc}}, {"rep", {makeFunction(*arena, stringType, {}, {}, {numberType}, {}, {stringType})}}, {"reverse", {stringToStringType}}, {"sub", {makeFunction(*arena, stringType, {}, {}, {numberType, optionalNumber}, {}, {stringType})}}, {"upper", {stringToStringType}}, {"split", {makeFunction(*arena, stringType, {}, {}, {optionalString}, {}, {arena->addType(TableTypeVar{{}, TableIndexer{numberType, stringType}, TypeLevel{}, TableState::Sealed})})}}, {"pack", {arena->addType(FunctionTypeVar{ arena->addTypePack(TypePack{{stringType}, anyTypePack}), oneStringPack, })}}, {"packsize", {makeFunction(*arena, stringType, {}, {}, {}, {}, {numberType})}}, {"unpack", {arena->addType(FunctionTypeVar{ arena->addTypePack(TypePack{{stringType, stringType, optionalNumber}}), anyTypePack, })}}, }; assignPropDocumentationSymbols(stringLib, "@luau/global/string"); TypeId tableType = arena->addType(TableTypeVar{std::move(stringLib), std::nullopt, TypeLevel{}, TableState::Sealed}); if (TableTypeVar* ttv = getMutable(tableType)) ttv->name = "string"; return arena->addType(TableTypeVar{{{{"__index", {tableType}}}}, std::nullopt, TypeLevel{}, TableState::Sealed}); } TypeId SingletonTypes::errorRecoveryType() { return errorType; } TypePackId SingletonTypes::errorRecoveryTypePack() { return errorTypePack; } TypeId SingletonTypes::errorRecoveryType(TypeId guess) { return guess; } TypePackId SingletonTypes::errorRecoveryTypePack(TypePackId guess) { return guess; } SingletonTypes& DEPRECATED_getSingletonTypes() { static SingletonTypes singletonTypes; return singletonTypes; } void persist(TypeId ty) { std::deque queue{ty}; while (!queue.empty()) { TypeId t = queue.front(); queue.pop_front(); if (t->persistent) continue; asMutable(t)->persistent = true; asMutable(t)->normal = true; // all persistent types are assumed to be normal if (auto btv = get(t)) queue.push_back(btv->boundTo); else if (auto ftv = get(t)) { persist(ftv->argTypes); persist(ftv->retTypes); } else if (auto ttv = get(t)) { LUAU_ASSERT(ttv->state != TableState::Free && ttv->state != TableState::Unsealed); for (const auto& [_name, prop] : ttv->props) queue.push_back(prop.type); if (ttv->indexer) { queue.push_back(ttv->indexer->indexType); queue.push_back(ttv->indexer->indexResultType); } } else if (auto ctv = get(t)) { for (const auto& [_name, prop] : ctv->props) queue.push_back(prop.type); } else if (auto utv = get(t)) { for (TypeId opt : utv->options) queue.push_back(opt); } else if (auto itv = get(t)) { for (TypeId opt : itv->parts) queue.push_back(opt); } else if (auto ctv = get(t)) { for (TypeId opt : ctv->parts) queue.push_back(opt); } else if (auto mtv = get(t)) { queue.push_back(mtv->table); queue.push_back(mtv->metatable); } else if (get(t) || get(t) || get(t) || get(t) || get(t)) { } else { LUAU_ASSERT(!"TypeId is not supported in a persist call"); } } } void persist(TypePackId tp) { if (tp->persistent) return; asMutable(tp)->persistent = true; if (auto p = get(tp)) { for (TypeId ty : p->head) persist(ty); if (p->tail) persist(*p->tail); } else if (auto vtp = get(tp)) { persist(vtp->ty); } else if (get(tp)) { } else { LUAU_ASSERT(!"TypePackId is not supported in a persist call"); } } const TypeLevel* getLevel(TypeId ty) { ty = follow(ty); if (auto ftv = get(ty)) return &ftv->level; else if (auto ttv = get(ty)) return &ttv->level; else if (auto ftv = get(ty)) return &ftv->level; else if (auto ctv = get(ty)) return &ctv->level; else return nullptr; } TypeLevel* getMutableLevel(TypeId ty) { return const_cast(getLevel(ty)); } std::optional getLevel(TypePackId tp) { tp = follow(tp); if (auto ftv = get(tp)) return ftv->level; else return std::nullopt; } const Property* lookupClassProp(const ClassTypeVar* cls, const Name& name) { while (cls) { auto it = cls->props.find(name); if (it != cls->props.end()) return &it->second; if (cls->parent) cls = get(*cls->parent); else return nullptr; LUAU_ASSERT(cls); } return nullptr; } bool isSubclass(const ClassTypeVar* cls, const ClassTypeVar* parent) { while (cls) { if (cls == parent) return true; else if (!cls->parent) return false; cls = get(*cls->parent); LUAU_ASSERT(cls); } return false; } const std::vector& getTypes(const UnionTypeVar* utv) { return utv->options; } const std::vector& getTypes(const IntersectionTypeVar* itv) { return itv->parts; } const std::vector& getTypes(const ConstrainedTypeVar* ctv) { return ctv->parts; } UnionTypeVarIterator begin(const UnionTypeVar* utv) { return UnionTypeVarIterator{utv}; } UnionTypeVarIterator end(const UnionTypeVar* utv) { return UnionTypeVarIterator{}; } IntersectionTypeVarIterator begin(const IntersectionTypeVar* itv) { return IntersectionTypeVarIterator{itv}; } IntersectionTypeVarIterator end(const IntersectionTypeVar* itv) { return IntersectionTypeVarIterator{}; } ConstrainedTypeVarIterator begin(const ConstrainedTypeVar* ctv) { return ConstrainedTypeVarIterator{ctv}; } ConstrainedTypeVarIterator end(const ConstrainedTypeVar* ctv) { return ConstrainedTypeVarIterator{}; } static std::vector parseFormatString(TypeChecker& typechecker, const char* data, size_t size) { const char* options = "cdiouxXeEfgGqs*"; std::vector result; for (size_t i = 0; i < size; ++i) { if (data[i] == '%') { i++; if (i < size && data[i] == '%') continue; // we just ignore all characters (including flags/precision) up until first alphabetic character while (i < size && !(data[i] > 0 && (isalpha(data[i]) || data[i] == '*'))) i++; if (i == size) break; if (data[i] == 'q' || data[i] == 's') result.push_back(typechecker.stringType); else if (data[i] == '*') result.push_back(typechecker.unknownType); else if (strchr(options, data[i])) result.push_back(typechecker.numberType); else result.push_back(typechecker.errorRecoveryType(typechecker.anyType)); } } return result; } std::optional> magicFunctionFormat( TypeChecker& typechecker, const ScopePtr& scope, const AstExprCall& expr, WithPredicate withPredicate) { auto [paramPack, _predicates] = withPredicate; TypeArena& arena = typechecker.currentModule->internalTypes; AstExprConstantString* fmt = nullptr; if (auto index = expr.func->as(); index && expr.self) { if (auto group = index->expr->as()) fmt = group->expr->as(); else fmt = index->expr->as(); } if (!expr.self && expr.args.size > 0) fmt = expr.args.data[0]->as(); if (!fmt) return std::nullopt; std::vector expected = parseFormatString(typechecker, fmt->value.data, fmt->value.size); const auto& [params, tail] = flatten(paramPack); size_t paramOffset = 1; size_t dataOffset = expr.self ? 0 : 1; // unify the prefix one argument at a time for (size_t i = 0; i < expected.size() && i + paramOffset < params.size(); ++i) { Location location = expr.args.data[std::min(i + dataOffset, expr.args.size - 1)]->location; typechecker.unify(params[i + paramOffset], expected[i], scope, location); } // if we know the argument count or if we have too many arguments for sure, we can issue an error if (FFlag::LuauStringFormatArgumentErrorFix) { size_t numActualParams = params.size(); size_t numExpectedParams = expected.size() + 1; // + 1 for the format string if (numExpectedParams != numActualParams && (!tail || numExpectedParams < numActualParams)) typechecker.reportError(TypeError{expr.location, CountMismatch{numExpectedParams, std::nullopt, numActualParams}}); } else { size_t actualParamSize = params.size() - paramOffset; if (expected.size() != actualParamSize && (!tail || expected.size() < actualParamSize)) typechecker.reportError(TypeError{expr.location, CountMismatch{expected.size(), std::nullopt, actualParamSize}}); } return WithPredicate{arena.addTypePack({typechecker.stringType})}; } static std::vector parsePatternString(TypeChecker& typechecker, const char* data, size_t size) { std::vector result; int depth = 0; bool parsingSet = false; for (size_t i = 0; i < size; ++i) { if (data[i] == '%') { ++i; if (!parsingSet && i < size && data[i] == 'b') i += 2; } else if (!parsingSet && data[i] == '[') { parsingSet = true; if (i + 1 < size && data[i + 1] == ']') i += 1; } else if (parsingSet && data[i] == ']') { parsingSet = false; } else if (data[i] == '(') { if (parsingSet) continue; if (i + 1 < size && data[i + 1] == ')') { i++; result.push_back(typechecker.numberType); continue; } ++depth; result.push_back(typechecker.stringType); } else if (data[i] == ')') { if (parsingSet) continue; --depth; if (depth < 0) break; } } if (depth != 0 || parsingSet) return std::vector(); if (result.empty()) result.push_back(typechecker.stringType); return result; } static std::optional> magicFunctionGmatch( TypeChecker& typechecker, const ScopePtr& scope, const AstExprCall& expr, WithPredicate withPredicate) { auto [paramPack, _predicates] = withPredicate; const auto& [params, tail] = flatten(paramPack); if (params.size() != 2) return std::nullopt; TypeArena& arena = typechecker.currentModule->internalTypes; AstExprConstantString* pattern = nullptr; size_t index = expr.self ? 0 : 1; if (expr.args.size > index) pattern = expr.args.data[index]->as(); if (!pattern) return std::nullopt; std::vector returnTypes = parsePatternString(typechecker, pattern->value.data, pattern->value.size); if (returnTypes.empty()) return std::nullopt; typechecker.unify(params[0], typechecker.stringType, scope, expr.args.data[0]->location); const TypePackId emptyPack = arena.addTypePack({}); const TypePackId returnList = arena.addTypePack(returnTypes); const TypeId iteratorType = arena.addType(FunctionTypeVar{emptyPack, returnList}); return WithPredicate{arena.addTypePack({iteratorType})}; } static std::optional> magicFunctionMatch( TypeChecker& typechecker, const ScopePtr& scope, const AstExprCall& expr, WithPredicate withPredicate) { auto [paramPack, _predicates] = withPredicate; const auto& [params, tail] = flatten(paramPack); if (params.size() < 2 || params.size() > 3) return std::nullopt; TypeArena& arena = typechecker.currentModule->internalTypes; AstExprConstantString* pattern = nullptr; size_t patternIndex = expr.self ? 0 : 1; if (expr.args.size > patternIndex) pattern = expr.args.data[patternIndex]->as(); if (!pattern) return std::nullopt; std::vector returnTypes = parsePatternString(typechecker, pattern->value.data, pattern->value.size); if (returnTypes.empty()) return std::nullopt; typechecker.unify(params[0], typechecker.stringType, scope, expr.args.data[0]->location); const TypeId optionalNumber = arena.addType(UnionTypeVar{{typechecker.nilType, typechecker.numberType}}); size_t initIndex = expr.self ? 1 : 2; if (params.size() == 3 && expr.args.size > initIndex) typechecker.unify(params[2], optionalNumber, scope, expr.args.data[initIndex]->location); const TypePackId returnList = arena.addTypePack(returnTypes); return WithPredicate{returnList}; } static std::optional> magicFunctionFind( TypeChecker& typechecker, const ScopePtr& scope, const AstExprCall& expr, WithPredicate withPredicate) { auto [paramPack, _predicates] = withPredicate; const auto& [params, tail] = flatten(paramPack); if (params.size() < 2 || params.size() > 4) return std::nullopt; TypeArena& arena = typechecker.currentModule->internalTypes; AstExprConstantString* pattern = nullptr; size_t patternIndex = expr.self ? 0 : 1; if (expr.args.size > patternIndex) pattern = expr.args.data[patternIndex]->as(); if (!pattern) return std::nullopt; bool plain = false; size_t plainIndex = expr.self ? 2 : 3; if (expr.args.size > plainIndex) { AstExprConstantBool* p = expr.args.data[plainIndex]->as(); plain = p && p->value; } std::vector returnTypes; if (!plain) { returnTypes = parsePatternString(typechecker, pattern->value.data, pattern->value.size); if (returnTypes.empty()) return std::nullopt; } typechecker.unify(params[0], typechecker.stringType, scope, expr.args.data[0]->location); const TypeId optionalNumber = arena.addType(UnionTypeVar{{typechecker.nilType, typechecker.numberType}}); const TypeId optionalBoolean = arena.addType(UnionTypeVar{{typechecker.nilType, typechecker.booleanType}}); size_t initIndex = expr.self ? 1 : 2; if (params.size() >= 3 && expr.args.size > initIndex) typechecker.unify(params[2], optionalNumber, scope, expr.args.data[initIndex]->location); if (params.size() == 4 && expr.args.size > plainIndex) typechecker.unify(params[3], optionalBoolean, scope, expr.args.data[plainIndex]->location); returnTypes.insert(returnTypes.begin(), {optionalNumber, optionalNumber}); const TypePackId returnList = arena.addTypePack(returnTypes); return WithPredicate{returnList}; } std::vector filterMap(TypeId type, TypeIdPredicate predicate) { type = follow(type); if (auto utv = get(type)) { std::set options; for (TypeId option : utv) if (auto out = predicate(follow(option))) options.insert(*out); return std::vector(options.begin(), options.end()); } else if (auto out = predicate(type)) return {*out}; return {}; } static Tags* getTags(TypeId ty) { ty = follow(ty); if (auto ftv = getMutable(ty)) return &ftv->tags; else if (auto ttv = getMutable(ty)) return &ttv->tags; else if (auto ctv = getMutable(ty)) return &ctv->tags; return nullptr; } void attachTag(TypeId ty, const std::string& tagName) { if (auto tags = getTags(ty)) tags->push_back(tagName); else LUAU_ASSERT(!"This TypeId does not support tags"); } void attachTag(Property& prop, const std::string& tagName) { prop.tags.push_back(tagName); } // We would ideally not expose this because it could cause a footgun. // If the Base class has a tag and you ask if Derived has that tag, it would return false. // Unfortunately, there's already use cases that's hard to disentangle. For now, we expose it. bool hasTag(const Tags& tags, const std::string& tagName) { return std::find(tags.begin(), tags.end(), tagName) != tags.end(); } bool hasTag(TypeId ty, const std::string& tagName) { ty = follow(ty); // We special case classes because getTags only returns a pointer to one vector of tags. // But classes has multiple vector of tags, represented throughout the hierarchy. if (auto ctv = get(ty)) { while (ctv) { if (hasTag(ctv->tags, tagName)) return true; else if (!ctv->parent) return false; ctv = get(*ctv->parent); LUAU_ASSERT(ctv); } } else if (auto tags = getTags(ty)) return hasTag(*tags, tagName); return false; } bool hasTag(const Property& prop, const std::string& tagName) { return hasTag(prop.tags, tagName); } bool TypeFun::operator==(const TypeFun& rhs) const { return type == rhs.type && typeParams == rhs.typeParams && typePackParams == rhs.typePackParams; } bool GenericTypeDefinition::operator==(const GenericTypeDefinition& rhs) const { return ty == rhs.ty && defaultValue == rhs.defaultValue; } bool GenericTypePackDefinition::operator==(const GenericTypePackDefinition& rhs) const { return tp == rhs.tp && defaultValue == rhs.defaultValue; } } // namespace Luau