luau/Analysis/src/Type.cpp

1607 lines
47 KiB
C++
Raw Normal View History

// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
#include "Luau/Type.h"
#include "Luau/BuiltinDefinitions.h"
#include "Luau/Common.h"
#include "Luau/ConstraintSolver.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/VisitType.h"
#include <algorithm>
#include <optional>
#include <stdexcept>
#include <unordered_map>
#include <unordered_set>
LUAU_FASTFLAG(DebugLuauFreezeArena)
LUAU_FASTINTVARIABLE(LuauTypeMaximumStringifierLength, 500)
LUAU_FASTINTVARIABLE(LuauTableTypeMaximumStringifierLength, 0)
LUAU_FASTINT(LuauTypeInferRecursionLimit)
2022-07-08 02:22:39 +01:00
LUAU_FASTFLAG(LuauUnknownAndNeverType)
LUAU_FASTFLAGVARIABLE(LuauMaybeGenericIntersectionTypes, false)
LUAU_FASTFLAG(LuauInstantiateInSubtyping)
namespace Luau
{
2022-06-17 02:05:14 +01:00
std::optional<WithPredicate<TypePackId>> magicFunctionFormat(
TypeChecker& typechecker, const ScopePtr& scope, const AstExprCall& expr, WithPredicate<TypePackId> withPredicate);
static bool dcrMagicFunctionFormat(MagicFunctionCallContext context);
2022-07-08 02:22:39 +01:00
static std::optional<WithPredicate<TypePackId>> magicFunctionGmatch(
TypeChecker& typechecker, const ScopePtr& scope, const AstExprCall& expr, WithPredicate<TypePackId> withPredicate);
static bool dcrMagicFunctionGmatch(MagicFunctionCallContext context);
2022-07-08 02:22:39 +01:00
2022-07-14 23:52:26 +01:00
static std::optional<WithPredicate<TypePackId>> magicFunctionMatch(
TypeChecker& typechecker, const ScopePtr& scope, const AstExprCall& expr, WithPredicate<TypePackId> withPredicate);
static bool dcrMagicFunctionMatch(MagicFunctionCallContext context);
2022-07-14 23:52:26 +01:00
static std::optional<WithPredicate<TypePackId>> magicFunctionFind(
TypeChecker& typechecker, const ScopePtr& scope, const AstExprCall& expr, WithPredicate<TypePackId> withPredicate);
static bool dcrMagicFunctionFind(MagicFunctionCallContext context);
2022-07-14 23:52:26 +01:00
TypeId follow(TypeId t)
{
return follow(t, [](TypeId t) {
return t;
});
}
TypeId follow(TypeId t, std::function<TypeId(TypeId)> mapper)
{
auto advance = [&mapper](TypeId ty) -> std::optional<TypeId> {
if (auto btv = get<Unifiable::Bound<TypeId>>(mapper(ty)))
return btv->boundTo;
else if (auto ttv = get<TableType>(mapper(ty)))
return ttv->boundTo;
else
return std::nullopt;
};
auto force = [&mapper](TypeId ty) {
if (auto ltv = get_if<LazyType>(&mapper(ty)->ty))
{
TypeId res = ltv->thunk();
if (get<LazyType>(res))
throw InternalCompilerError("Lazy Type cannot resolve to another Lazy Type");
*asMutable(ty) = BoundType(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 InternalCompilerError("Luau::follow detected a Type cycle!!");
}
}
}
std::vector<TypeId> flattenIntersection(TypeId ty)
{
if (!get<IntersectionType>(follow(ty)))
return {ty};
std::unordered_set<TypeId> seen;
std::deque<TypeId> queue{ty};
std::vector<TypeId> 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<IntersectionType>(current))
{
for (TypeId ty : itv->parts)
queue.push_back(ty);
}
else
result.push_back(current);
}
return result;
}
bool isPrim(TypeId ty, PrimitiveType::Type primType)
{
auto p = get<PrimitiveType>(follow(ty));
return p && p->type == primType;
}
bool isNil(TypeId ty)
{
return isPrim(ty, PrimitiveType::NilType);
}
bool isBoolean(TypeId ty)
{
if (isPrim(ty, PrimitiveType::Boolean) || get<BooleanSingleton>(get<SingletonType>(follow(ty))))
2022-02-24 23:53:37 +00:00
return true;
if (auto utv = get<UnionType>(follow(ty)))
2022-02-24 23:53:37 +00:00
return std::all_of(begin(utv), end(utv), isBoolean);
2022-02-24 23:53:37 +00:00
return false;
}
bool isNumber(TypeId ty)
{
return isPrim(ty, PrimitiveType::Number);
}
2022-03-04 16:36:33 +00:00
// Returns true when ty is a subtype of string
bool isString(TypeId ty)
{
2022-07-14 23:52:26 +01:00
ty = follow(ty);
if (isPrim(ty, PrimitiveType::String) || get<StringSingleton>(get<SingletonType>(ty)))
2022-02-24 23:53:37 +00:00
return true;
if (auto utv = get<UnionType>(ty))
2022-02-24 23:53:37 +00:00
return std::all_of(begin(utv), end(utv), isString);
2022-02-24 23:53:37 +00:00
return false;
}
2022-03-04 16:36:33 +00:00
// Returns true when ty is a supertype of string
bool maybeString(TypeId ty)
{
2022-06-24 02:56:00 +01:00
ty = follow(ty);
2022-04-15 00:57:43 +01:00
if (isPrim(ty, PrimitiveType::String) || get<AnyType>(ty))
2022-07-14 23:52:26 +01:00
return true;
2022-03-04 16:36:33 +00:00
if (auto utv = get<UnionType>(ty))
2022-06-24 02:56:00 +01:00
return std::any_of(begin(utv), end(utv), maybeString);
2022-03-04 16:36:33 +00:00
2022-06-24 02:56:00 +01:00
return false;
2022-03-04 16:36:33 +00:00
}
bool isThread(TypeId ty)
{
return isPrim(ty, PrimitiveType::Thread);
}
bool isOptional(TypeId ty)
{
if (isNil(ty))
return true;
2022-03-18 00:46:04 +00:00
ty = follow(ty);
if (get<AnyType>(ty) || (FFlag::LuauUnknownAndNeverType && get<UnknownType>(ty)))
2022-03-18 00:46:04 +00:00
return true;
auto utv = get<UnionType>(ty);
2022-02-24 23:53:37 +00:00
if (!utv)
return false;
2022-02-24 23:53:37 +00:00
2022-05-20 01:02:24 +01:00
return std::any_of(begin(utv), end(utv), isOptional);
}
bool isTableIntersection(TypeId ty)
{
if (!get<IntersectionType>(follow(ty)))
return false;
std::vector<TypeId> parts = flattenIntersection(ty);
return std::all_of(parts.begin(), parts.end(), getTableType);
}
bool isOverloadedFunction(TypeId ty)
{
if (!get<IntersectionType>(follow(ty)))
return false;
auto isFunction = [](TypeId part) -> bool {
return get<FunctionType>(part);
};
std::vector<TypeId> parts = flattenIntersection(ty);
return std::all_of(parts.begin(), parts.end(), isFunction);
}
std::optional<TypeId> getMetatable(TypeId type, NotNull<BuiltinTypes> builtinTypes)
{
2022-07-14 23:52:26 +01:00
type = follow(type);
if (const MetatableType* mtType = get<MetatableType>(type))
return mtType->metatable;
else if (const ClassType* classType = get<ClassType>(type))
return classType->metatable;
2022-02-24 23:53:37 +00:00
else if (isString(type))
{
auto ptv = get<PrimitiveType>(builtinTypes->stringType);
2022-02-24 23:53:37 +00:00
LUAU_ASSERT(ptv && ptv->metatable);
return ptv->metatable;
}
2022-02-24 23:53:37 +00:00
return std::nullopt;
}
const TableType* getTableType(TypeId type)
{
2022-02-24 23:53:37 +00:00
type = follow(type);
2022-02-04 16:45:57 +00:00
if (const TableType* ttv = get<TableType>(type))
return ttv;
else if (const MetatableType* mtv = get<MetatableType>(type))
return get<TableType>(follow(mtv->table));
else
return nullptr;
}
TableType* getMutableTableType(TypeId type)
{
return const_cast<TableType*>(getTableType(type));
}
const std::string* getName(TypeId type)
{
type = follow(type);
if (auto mtv = get<MetatableType>(type))
{
if (mtv->syntheticName)
return &*mtv->syntheticName;
2022-02-24 23:53:37 +00:00
type = follow(mtv->table);
}
if (auto ttv = get<TableType>(type))
{
if (ttv->name)
return &*ttv->name;
if (ttv->syntheticName)
return &*ttv->syntheticName;
}
return nullptr;
}
2022-03-24 22:04:14 +00:00
std::optional<ModuleName> getDefinitionModuleName(TypeId type)
{
type = follow(type);
if (auto ttv = get<TableType>(type))
2022-03-24 22:04:14 +00:00
{
if (!ttv->definitionModuleName.empty())
return ttv->definitionModuleName;
}
else if (auto ftv = get<FunctionType>(type))
2022-03-24 22:04:14 +00:00
{
if (ftv->definition)
return ftv->definition->definitionModuleName;
}
else if (auto ctv = get<ClassType>(type))
2022-04-21 22:44:27 +01:00
{
if (!ctv->definitionModuleName.empty())
return ctv->definitionModuleName;
}
2022-03-24 22:04:14 +00:00
return std::nullopt;
}
bool isSubset(const UnionType& super, const UnionType& sub)
{
std::unordered_set<TypeId> 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<FunctionType>(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);
2022-07-08 02:22:39 +01:00
if (FFlag::LuauMaybeGenericIntersectionTypes)
{
ty = follow(ty);
if (get<FreeType>(ty))
2022-07-08 02:22:39 +01:00
return true;
if (auto ttv = get<TableType>(ty))
2022-07-08 02:22:39 +01:00
{
// TODO: recurse on table types CLI-39914
(void)ttv;
return true;
}
if (auto itv = get<IntersectionType>(ty))
2022-07-08 02:22:39 +01:00
{
return std::any_of(begin(itv), end(itv), maybeGeneric);
}
return isGeneric(ty);
}
ty = follow(ty);
if (get<FreeType>(ty))
return true;
else if (auto ttv = get<TableType>(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<SingletonType>(ty))
return true;
if (const UnionType* utv = get<UnionType>(ty))
for (TypeId option : utv)
if (get<SingletonType>(follow(option)))
return true;
return false;
}
bool hasLength(TypeId ty, DenseHashSet<TypeId>& seen, int* recursionCount)
{
2022-06-24 02:56:00 +01:00
RecursionLimiter _rl(recursionCount, FInt::LuauTypeInferRecursionLimit);
ty = follow(ty);
if (seen.contains(ty))
return true;
if (isString(ty) || get<AnyType>(ty) || get<TableType>(ty) || get<MetatableType>(ty))
return true;
if (auto uty = get<UnionType>(ty))
{
seen.insert(ty);
for (TypeId part : uty->options)
{
if (!hasLength(part, seen, recursionCount))
return false;
}
return true;
}
if (auto ity = get<IntersectionType>(ty))
{
seen.insert(ty);
for (TypeId part : ity->parts)
{
if (hasLength(part, seen, recursionCount))
return true;
}
return false;
}
return false;
}
BlockedType::BlockedType()
2022-06-17 02:05:14 +01:00
: index(++nextIndex)
{
}
int BlockedType::nextIndex = 0;
2022-06-17 02:05:14 +01:00
PendingExpansionType::PendingExpansionType(
std::optional<AstName> prefix, AstName name, std::vector<TypeId> typeArguments, std::vector<TypePackId> packArguments)
: prefix(prefix)
, name(name)
2022-08-04 23:35:33 +01:00
, typeArguments(typeArguments)
, packArguments(packArguments)
, index(++nextIndex)
{
}
size_t PendingExpansionType::nextIndex = 0;
2022-08-04 23:35:33 +01:00
FunctionType::FunctionType(TypePackId argTypes, TypePackId retTypes, std::optional<FunctionDefinition> defn, bool hasSelf)
: definition(std::move(defn))
, argTypes(argTypes)
2022-06-17 02:05:14 +01:00
, retTypes(retTypes)
, hasSelf(hasSelf)
{
}
FunctionType::FunctionType(TypeLevel level, TypePackId argTypes, TypePackId retTypes, std::optional<FunctionDefinition> defn, bool hasSelf)
: definition(std::move(defn))
, level(level)
, argTypes(argTypes)
2022-06-17 02:05:14 +01:00
, retTypes(retTypes)
, hasSelf(hasSelf)
{
}
FunctionType::FunctionType(
TypeLevel level, Scope* scope, TypePackId argTypes, TypePackId retTypes, std::optional<FunctionDefinition> defn, bool hasSelf)
: definition(std::move(defn))
, level(level)
, scope(scope)
, argTypes(argTypes)
, retTypes(retTypes)
, hasSelf(hasSelf)
{
}
FunctionType::FunctionType(std::vector<TypeId> generics, std::vector<TypePackId> genericPacks, TypePackId argTypes, TypePackId retTypes,
std::optional<FunctionDefinition> defn, bool hasSelf)
: definition(std::move(defn))
, generics(generics)
, genericPacks(genericPacks)
, argTypes(argTypes)
2022-06-17 02:05:14 +01:00
, retTypes(retTypes)
, hasSelf(hasSelf)
{
}
FunctionType::FunctionType(TypeLevel level, std::vector<TypeId> generics, std::vector<TypePackId> genericPacks, TypePackId argTypes,
2022-06-17 02:05:14 +01:00
TypePackId retTypes, std::optional<FunctionDefinition> defn, bool hasSelf)
: definition(std::move(defn))
, generics(generics)
, genericPacks(genericPacks)
, level(level)
, argTypes(argTypes)
2022-06-17 02:05:14 +01:00
, retTypes(retTypes)
, hasSelf(hasSelf)
{
}
FunctionType::FunctionType(TypeLevel level, Scope* scope, std::vector<TypeId> generics, std::vector<TypePackId> genericPacks, TypePackId argTypes,
TypePackId retTypes, std::optional<FunctionDefinition> defn, bool hasSelf)
: definition(std::move(defn))
, generics(generics)
, genericPacks(genericPacks)
, level(level)
, scope(scope)
, argTypes(argTypes)
, retTypes(retTypes)
, hasSelf(hasSelf)
{
}
TableType::TableType(TableState state, TypeLevel level, Scope* scope)
: state(state)
, level(level)
, scope(scope)
{
}
TableType::TableType(const Props& props, const std::optional<TableIndexer>& indexer, TypeLevel level, TableState state)
: props(props)
, indexer(indexer)
, state(state)
, level(level)
{
}
TableType::TableType(const Props& props, const std::optional<TableIndexer>& indexer, TypeLevel level, Scope* scope, TableState state)
: props(props)
, indexer(indexer)
, state(state)
, level(level)
, scope(scope)
{
}
// Test Types for equivalence
// More complex than we'd like because Types 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<void*>(lhs), const_cast<void*>(rhs));
if (seen.find(p) != seen.end())
return true;
seen.insert(p);
return false;
}
bool areEqual(SeenSet& seen, const FunctionType& lhs, const FunctionType& rhs)
{
if (areSeen(seen, &lhs, &rhs))
return true;
// TODO: check generics CLI-39915
if (!areEqual(seen, *lhs.argTypes, *rhs.argTypes))
return false;
2022-06-17 02:05:14 +01:00
if (!areEqual(seen, *lhs.retTypes, *rhs.retTypes))
return false;
return true;
}
bool areEqual(SeenSet& seen, const TableType& lhs, const TableType& 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 MetatableType& lhs, const MetatableType& rhs)
{
2022-02-18 01:18:01 +00:00
if (areSeen(seen, &lhs, &rhs))
2022-01-14 16:20:09 +00:00
return true;
return areEqual(seen, *lhs.table, *rhs.table) && areEqual(seen, *lhs.metatable, *rhs.metatable);
}
bool areEqual(SeenSet& seen, const Type& lhs, const Type& rhs)
{
if (auto bound = get_if<BoundType>(&lhs.ty))
return areEqual(seen, *bound->boundTo, rhs);
if (auto bound = get_if<BoundType>(&rhs.ty))
return areEqual(seen, lhs, *bound->boundTo);
if (lhs.ty.index() != rhs.ty.index())
return false;
{
const FreeType* lf = get_if<FreeType>(&lhs.ty);
const FreeType* rf = get_if<FreeType>(&rhs.ty);
if (lf && rf)
return lf->index == rf->index;
}
{
const GenericType* lg = get_if<GenericType>(&lhs.ty);
const GenericType* rg = get_if<GenericType>(&rhs.ty);
if (lg && rg)
return lg->index == rg->index;
}
{
const PrimitiveType* lp = get_if<PrimitiveType>(&lhs.ty);
const PrimitiveType* rp = get_if<PrimitiveType>(&rhs.ty);
if (lp && rp)
return lp->type == rp->type;
}
{
const GenericType* lg = get_if<GenericType>(&lhs.ty);
const GenericType* rg = get_if<GenericType>(&rhs.ty);
if (lg && rg)
return lg->index == rg->index;
}
{
const ErrorType* le = get_if<ErrorType>(&lhs.ty);
const ErrorType* re = get_if<ErrorType>(&rhs.ty);
if (le && re)
return le->index == re->index;
}
{
const FunctionType* lf = get_if<FunctionType>(&lhs.ty);
const FunctionType* rf = get_if<FunctionType>(&rhs.ty);
if (lf && rf)
return areEqual(seen, *lf, *rf);
}
{
const TableType* lt = get_if<TableType>(&lhs.ty);
const TableType* rt = get_if<TableType>(&rhs.ty);
if (lt && rt)
return areEqual(seen, *lt, *rt);
}
{
const MetatableType* lmt = get_if<MetatableType>(&lhs.ty);
const MetatableType* rmt = get_if<MetatableType>(&rhs.ty);
if (lmt && rmt)
return areEqual(seen, *lmt, *rmt);
}
if (get_if<AnyType>(&lhs.ty) && get_if<AnyType>(&rhs.ty))
return true;
return false;
}
Type* asMutable(TypeId ty)
{
return const_cast<Type*>(ty);
}
bool Type::operator==(const Type& rhs) const
{
SeenSet seen;
return areEqual(seen, *this, rhs);
}
bool Type::operator!=(const Type& rhs) const
{
SeenSet seen;
return !areEqual(seen, *this, rhs);
}
Type& Type::operator=(const TypeVariant& rhs)
{
ty = rhs;
return *this;
}
Type& Type::operator=(TypeVariant&& rhs)
{
ty = std::move(rhs);
return *this;
}
Type& Type::operator=(const Type& rhs)
{
2022-07-08 02:22:39 +01:00
LUAU_ASSERT(owningArena == rhs.owningArena);
LUAU_ASSERT(!rhs.persistent);
2022-07-08 02:22:39 +01:00
reassign(rhs);
return *this;
}
TypeId makeFunction(TypeArena& arena, std::optional<TypeId> selfType, std::initializer_list<TypeId> generics,
std::initializer_list<TypePackId> genericPacks, std::initializer_list<TypeId> paramTypes, std::initializer_list<std::string> paramNames,
std::initializer_list<TypeId> retTypes);
BuiltinTypes::BuiltinTypes()
: arena(new TypeArena)
, debugFreezeArena(FFlag::DebugLuauFreezeArena)
, nilType(arena->addType(Type{PrimitiveType{PrimitiveType::NilType}, /*persistent*/ true}))
, numberType(arena->addType(Type{PrimitiveType{PrimitiveType::Number}, /*persistent*/ true}))
, stringType(arena->addType(Type{PrimitiveType{PrimitiveType::String}, /*persistent*/ true}))
, booleanType(arena->addType(Type{PrimitiveType{PrimitiveType::Boolean}, /*persistent*/ true}))
, threadType(arena->addType(Type{PrimitiveType{PrimitiveType::Thread}, /*persistent*/ true}))
, functionType(arena->addType(Type{PrimitiveType{PrimitiveType::Function}, /*persistent*/ true}))
, classType(arena->addType(Type{ClassType{"class", {}, std::nullopt, std::nullopt, {}, {}, {}}, /*persistent*/ true}))
, trueType(arena->addType(Type{SingletonType{BooleanSingleton{true}}, /*persistent*/ true}))
, falseType(arena->addType(Type{SingletonType{BooleanSingleton{false}}, /*persistent*/ true}))
, anyType(arena->addType(Type{AnyType{}, /*persistent*/ true}))
, unknownType(arena->addType(Type{UnknownType{}, /*persistent*/ true}))
, neverType(arena->addType(Type{NeverType{}, /*persistent*/ true}))
, errorType(arena->addType(Type{ErrorType{}, /*persistent*/ true}))
, falsyType(arena->addType(Type{UnionType{{falseType, nilType}}, /*persistent*/ true}))
, truthyType(arena->addType(Type{NegationType{falsyType}, /*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 = PrimitiveType{PrimitiveType::String, stringMetatable};
persist(stringMetatable);
persist(uninhabitableTypePack);
freeze(*arena);
}
BuiltinTypes::~BuiltinTypes()
{
// 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 BuiltinTypes::makeStringMetatable()
{
const TypeId optionalNumber = arena->addType(UnionType{{nilType, numberType}});
const TypeId optionalString = arena->addType(UnionType{{nilType, stringType}});
const TypeId optionalBoolean = arena->addType(UnionType{{nilType, booleanType}});
const TypePackId oneStringPack = arena->addTypePack({stringType});
const TypePackId anyTypePack = arena->addTypePack(TypePackVar{VariadicTypePack{anyType}, true});
FunctionType formatFTV{arena->addTypePack(TypePack{{stringType}, anyTypePack}), oneStringPack};
formatFTV.magicFunction = &magicFunctionFormat;
const TypeId formatFn = arena->addType(formatFTV);
attachDcrMagicFunction(formatFn, dcrMagicFunctionFormat);
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(UnionType{{stringType, arena->addType(TableType({}, 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(FunctionType{emptyPack, stringVariadicList})});
2022-07-08 02:22:39 +01:00
attachMagicFunction(gmatchFunc, magicFunctionGmatch);
attachDcrMagicFunction(gmatchFunc, dcrMagicFunctionGmatch);
const TypeId matchFunc = arena->addType(
FunctionType{arena->addTypePack({stringType, stringType, optionalNumber}), arena->addTypePack(TypePackVar{VariadicTypePack{stringType}})});
2022-07-14 23:52:26 +01:00
attachMagicFunction(matchFunc, magicFunctionMatch);
attachDcrMagicFunction(matchFunc, dcrMagicFunctionMatch);
2022-07-14 23:52:26 +01:00
const TypeId findFunc = arena->addType(FunctionType{arena->addTypePack({stringType, stringType, optionalNumber, optionalBoolean}),
2022-07-14 23:52:26 +01:00
arena->addTypePack(TypePack{{optionalNumber, optionalNumber}, stringVariadicList})});
attachMagicFunction(findFunc, magicFunctionFind);
attachDcrMagicFunction(findFunc, dcrMagicFunctionFind);
2022-07-14 23:52:26 +01:00
TableType::Props stringLib = {
{"byte", {arena->addType(FunctionType{arena->addTypePack({stringType, optionalNumber, optionalNumber}), numberVariadicList})}},
{"char", {arena->addType(FunctionType{numberVariadicList, arena->addTypePack({stringType})})}},
2022-07-14 23:52:26 +01:00
{"find", {findFunc}},
{"format", {formatFn}}, // FIXME
{"gmatch", {gmatchFunc}},
{"gsub", {gsubFunc}},
{"len", {makeFunction(*arena, stringType, {}, {}, {}, {}, {numberType})}},
{"lower", {stringToStringType}},
2022-07-14 23:52:26 +01:00
{"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(TableType{{}, TableIndexer{numberType, stringType}, TypeLevel{}, TableState::Sealed})})}},
{"pack", {arena->addType(FunctionType{
arena->addTypePack(TypePack{{stringType}, anyTypePack}),
oneStringPack,
})}},
{"packsize", {makeFunction(*arena, stringType, {}, {}, {}, {}, {numberType})}},
{"unpack", {arena->addType(FunctionType{
arena->addTypePack(TypePack{{stringType, stringType, optionalNumber}}),
anyTypePack,
})}},
};
assignPropDocumentationSymbols(stringLib, "@luau/global/string");
TypeId tableType = arena->addType(TableType{std::move(stringLib), std::nullopt, TypeLevel{}, TableState::Sealed});
if (TableType* ttv = getMutable<TableType>(tableType))
ttv->name = "typeof(string)";
return arena->addType(TableType{{{{"__index", {tableType}}}}, std::nullopt, TypeLevel{}, TableState::Sealed});
}
TypeId BuiltinTypes::errorRecoveryType()
{
return errorType;
}
TypePackId BuiltinTypes::errorRecoveryTypePack()
{
return errorTypePack;
}
TypeId BuiltinTypes::errorRecoveryType(TypeId guess)
{
2022-04-29 02:24:24 +01:00
return guess;
}
TypePackId BuiltinTypes::errorRecoveryTypePack(TypePackId guess)
{
2022-04-29 02:24:24 +01:00
return guess;
}
void persist(TypeId ty)
{
std::deque<TypeId> queue{ty};
while (!queue.empty())
{
TypeId t = queue.front();
queue.pop_front();
if (t->persistent)
continue;
asMutable(t)->persistent = true;
if (auto btv = get<BoundType>(t))
queue.push_back(btv->boundTo);
else if (auto ftv = get<FunctionType>(t))
{
persist(ftv->argTypes);
2022-06-17 02:05:14 +01:00
persist(ftv->retTypes);
}
else if (auto ttv = get<TableType>(t))
{
2022-03-04 16:36:33 +00:00
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<ClassType>(t))
{
for (const auto& [_name, prop] : ctv->props)
queue.push_back(prop.type);
}
else if (auto utv = get<UnionType>(t))
{
for (TypeId opt : utv->options)
queue.push_back(opt);
}
else if (auto itv = get<IntersectionType>(t))
{
for (TypeId opt : itv->parts)
queue.push_back(opt);
}
else if (auto mtv = get<MetatableType>(t))
{
queue.push_back(mtv->table);
queue.push_back(mtv->metatable);
}
else if (get<GenericType>(t) || get<AnyType>(t) || get<FreeType>(t) || get<SingletonType>(t) || get<PrimitiveType>(t) || get<NegationType>(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<TypePack>(tp))
{
for (TypeId ty : p->head)
persist(ty);
if (p->tail)
persist(*p->tail);
}
else if (auto vtp = get<VariadicTypePack>(tp))
{
persist(vtp->ty);
}
else if (get<GenericTypePack>(tp))
{
}
else
{
LUAU_ASSERT(!"TypePackId is not supported in a persist call");
}
}
const TypeLevel* getLevel(TypeId ty)
{
ty = follow(ty);
if (auto ftv = get<Unifiable::Free>(ty))
return &ftv->level;
else if (auto ttv = get<TableType>(ty))
return &ttv->level;
else if (auto ftv = get<FunctionType>(ty))
return &ftv->level;
else
return nullptr;
}
TypeLevel* getMutableLevel(TypeId ty)
{
return const_cast<TypeLevel*>(getLevel(ty));
}
2022-04-15 00:57:43 +01:00
std::optional<TypeLevel> getLevel(TypePackId tp)
{
tp = follow(tp);
if (auto ftv = get<Unifiable::Free>(tp))
return ftv->level;
else
return std::nullopt;
}
const Property* lookupClassProp(const ClassType* 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<ClassType>(*cls->parent);
else
return nullptr;
LUAU_ASSERT(cls);
}
return nullptr;
}
bool isSubclass(const ClassType* cls, const ClassType* parent)
{
while (cls)
{
if (cls == parent)
return true;
else if (!cls->parent)
return false;
cls = get<ClassType>(*cls->parent);
LUAU_ASSERT(cls);
}
return false;
}
const std::vector<TypeId>& getTypes(const UnionType* utv)
{
2022-07-08 02:22:39 +01:00
return utv->options;
}
const std::vector<TypeId>& getTypes(const IntersectionType* itv)
{
2022-07-08 02:22:39 +01:00
return itv->parts;
}
UnionTypeIterator begin(const UnionType* utv)
{
return UnionTypeIterator{utv};
}
UnionTypeIterator end(const UnionType* utv)
{
return UnionTypeIterator{};
}
IntersectionTypeIterator begin(const IntersectionType* itv)
{
return IntersectionTypeIterator{itv};
}
IntersectionTypeIterator end(const IntersectionType* itv)
{
return IntersectionTypeIterator{};
}
static std::vector<TypeId> parseFormatString(NotNull<BuiltinTypes> builtinTypes, const char* data, size_t size)
{
const char* options = "cdiouxXeEfgGqs*";
std::vector<TypeId> 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(builtinTypes->stringType);
else if (data[i] == '*')
result.push_back(builtinTypes->unknownType);
else if (strchr(options, data[i]))
result.push_back(builtinTypes->numberType);
else
result.push_back(builtinTypes->errorRecoveryType(builtinTypes->anyType));
}
}
return result;
}
2022-06-17 02:05:14 +01:00
std::optional<WithPredicate<TypePackId>> magicFunctionFormat(
TypeChecker& typechecker, const ScopePtr& scope, const AstExprCall& expr, WithPredicate<TypePackId> withPredicate)
{
2022-06-17 02:05:14 +01:00
auto [paramPack, _predicates] = withPredicate;
TypeArena& arena = typechecker.currentModule->internalTypes;
AstExprConstantString* fmt = nullptr;
if (auto index = expr.func->as<AstExprIndexName>(); index && expr.self)
{
if (auto group = index->expr->as<AstExprGroup>())
fmt = group->expr->as<AstExprConstantString>();
else
fmt = index->expr->as<AstExprConstantString>();
}
if (!expr.self && expr.args.size > 0)
fmt = expr.args.data[0]->as<AstExprConstantString>();
if (!fmt)
return std::nullopt;
std::vector<TypeId> expected = parseFormatString(typechecker.builtinTypes, 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
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}});
2022-06-17 02:05:14 +01:00
return WithPredicate<TypePackId>{arena.addTypePack({typechecker.stringType})};
}
static bool dcrMagicFunctionFormat(MagicFunctionCallContext context)
{
TypeArena* arena = context.solver->arena;
AstExprConstantString* fmt = nullptr;
if (auto index = context.callSite->func->as<AstExprIndexName>(); index && context.callSite->self)
{
if (auto group = index->expr->as<AstExprGroup>())
fmt = group->expr->as<AstExprConstantString>();
else
fmt = index->expr->as<AstExprConstantString>();
}
if (!context.callSite->self && context.callSite->args.size > 0)
fmt = context.callSite->args.data[0]->as<AstExprConstantString>();
if (!fmt)
return false;
std::vector<TypeId> expected = parseFormatString(context.solver->builtinTypes, fmt->value.data, fmt->value.size);
const auto& [params, tail] = flatten(context.arguments);
size_t paramOffset = 1;
// unify the prefix one argument at a time
for (size_t i = 0; i < expected.size() && i + paramOffset < params.size(); ++i)
{
context.solver->unify(params[i + paramOffset], expected[i], context.solver->rootScope);
}
// if we know the argument count or if we have too many arguments for sure, we can issue an error
size_t numActualParams = params.size();
size_t numExpectedParams = expected.size() + 1; // + 1 for the format string
if (numExpectedParams != numActualParams && (!tail || numExpectedParams < numActualParams))
context.solver->reportError(TypeError{context.callSite->location, CountMismatch{numExpectedParams, std::nullopt, numActualParams}});
TypePackId resultPack = arena->addTypePack({context.solver->builtinTypes->stringType});
asMutable(context.result)->ty.emplace<BoundTypePack>(resultPack);
return true;
}
static std::vector<TypeId> parsePatternString(NotNull<BuiltinTypes> builtinTypes, const char* data, size_t size)
2022-07-08 02:22:39 +01:00
{
std::vector<TypeId> 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(builtinTypes->numberType);
2022-07-08 02:22:39 +01:00
continue;
}
++depth;
result.push_back(builtinTypes->stringType);
2022-07-08 02:22:39 +01:00
}
else if (data[i] == ')')
{
if (parsingSet)
continue;
--depth;
if (depth < 0)
break;
}
}
if (depth != 0 || parsingSet)
return std::vector<TypeId>();
if (result.empty())
result.push_back(builtinTypes->stringType);
2022-07-08 02:22:39 +01:00
return result;
}
static std::optional<WithPredicate<TypePackId>> magicFunctionGmatch(
TypeChecker& typechecker, const ScopePtr& scope, const AstExprCall& expr, WithPredicate<TypePackId> 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<AstExprConstantString>();
if (!pattern)
return std::nullopt;
std::vector<TypeId> returnTypes = parsePatternString(typechecker.builtinTypes, pattern->value.data, pattern->value.size);
2022-07-08 02:22:39 +01:00
if (returnTypes.empty())
return std::nullopt;
typechecker.unify(params[0], typechecker.stringType, scope, expr.args.data[0]->location);
2022-07-08 02:22:39 +01:00
const TypePackId emptyPack = arena.addTypePack({});
const TypePackId returnList = arena.addTypePack(returnTypes);
const TypeId iteratorType = arena.addType(FunctionType{emptyPack, returnList});
2022-07-08 02:22:39 +01:00
return WithPredicate<TypePackId>{arena.addTypePack({iteratorType})};
}
static bool dcrMagicFunctionGmatch(MagicFunctionCallContext context)
{
const auto& [params, tail] = flatten(context.arguments);
if (params.size() != 2)
return false;
TypeArena* arena = context.solver->arena;
AstExprConstantString* pattern = nullptr;
size_t index = context.callSite->self ? 0 : 1;
if (context.callSite->args.size > index)
pattern = context.callSite->args.data[index]->as<AstExprConstantString>();
if (!pattern)
return false;
std::vector<TypeId> returnTypes = parsePatternString(context.solver->builtinTypes, pattern->value.data, pattern->value.size);
if (returnTypes.empty())
return false;
context.solver->unify(params[0], context.solver->builtinTypes->stringType, context.solver->rootScope);
const TypePackId emptyPack = arena->addTypePack({});
const TypePackId returnList = arena->addTypePack(returnTypes);
const TypeId iteratorType = arena->addType(FunctionType{emptyPack, returnList});
const TypePackId resTypePack = arena->addTypePack({iteratorType});
asMutable(context.result)->ty.emplace<BoundTypePack>(resTypePack);
return true;
}
2022-07-14 23:52:26 +01:00
static std::optional<WithPredicate<TypePackId>> magicFunctionMatch(
TypeChecker& typechecker, const ScopePtr& scope, const AstExprCall& expr, WithPredicate<TypePackId> 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<AstExprConstantString>();
if (!pattern)
return std::nullopt;
std::vector<TypeId> returnTypes = parsePatternString(typechecker.builtinTypes, pattern->value.data, pattern->value.size);
2022-07-14 23:52:26 +01:00
if (returnTypes.empty())
return std::nullopt;
typechecker.unify(params[0], typechecker.stringType, scope, expr.args.data[0]->location);
2022-07-14 23:52:26 +01:00
const TypeId optionalNumber = arena.addType(UnionType{{typechecker.nilType, typechecker.numberType}});
2022-07-14 23:52:26 +01:00
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);
2022-07-14 23:52:26 +01:00
const TypePackId returnList = arena.addTypePack(returnTypes);
return WithPredicate<TypePackId>{returnList};
}
static bool dcrMagicFunctionMatch(MagicFunctionCallContext context)
{
const auto& [params, tail] = flatten(context.arguments);
if (params.size() < 2 || params.size() > 3)
return false;
TypeArena* arena = context.solver->arena;
AstExprConstantString* pattern = nullptr;
size_t patternIndex = context.callSite->self ? 0 : 1;
if (context.callSite->args.size > patternIndex)
pattern = context.callSite->args.data[patternIndex]->as<AstExprConstantString>();
if (!pattern)
return false;
std::vector<TypeId> returnTypes = parsePatternString(context.solver->builtinTypes, pattern->value.data, pattern->value.size);
if (returnTypes.empty())
return false;
context.solver->unify(params[0], context.solver->builtinTypes->stringType, context.solver->rootScope);
const TypeId optionalNumber = arena->addType(UnionType{{context.solver->builtinTypes->nilType, context.solver->builtinTypes->numberType}});
size_t initIndex = context.callSite->self ? 1 : 2;
if (params.size() == 3 && context.callSite->args.size > initIndex)
context.solver->unify(params[2], optionalNumber, context.solver->rootScope);
const TypePackId returnList = arena->addTypePack(returnTypes);
asMutable(context.result)->ty.emplace<BoundTypePack>(returnList);
return true;
}
2022-07-14 23:52:26 +01:00
static std::optional<WithPredicate<TypePackId>> magicFunctionFind(
TypeChecker& typechecker, const ScopePtr& scope, const AstExprCall& expr, WithPredicate<TypePackId> 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<AstExprConstantString>();
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<AstExprConstantBool>();
plain = p && p->value;
}
std::vector<TypeId> returnTypes;
if (!plain)
{
returnTypes = parsePatternString(typechecker.builtinTypes, pattern->value.data, pattern->value.size);
2022-07-14 23:52:26 +01:00
if (returnTypes.empty())
return std::nullopt;
}
typechecker.unify(params[0], typechecker.stringType, scope, expr.args.data[0]->location);
2022-07-14 23:52:26 +01:00
const TypeId optionalNumber = arena.addType(UnionType{{typechecker.nilType, typechecker.numberType}});
const TypeId optionalBoolean = arena.addType(UnionType{{typechecker.nilType, typechecker.booleanType}});
2022-07-14 23:52:26 +01:00
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);
2022-07-14 23:52:26 +01:00
if (params.size() == 4 && expr.args.size > plainIndex)
typechecker.unify(params[3], optionalBoolean, scope, expr.args.data[plainIndex]->location);
2022-07-14 23:52:26 +01:00
returnTypes.insert(returnTypes.begin(), {optionalNumber, optionalNumber});
const TypePackId returnList = arena.addTypePack(returnTypes);
return WithPredicate<TypePackId>{returnList};
}
static bool dcrMagicFunctionFind(MagicFunctionCallContext context)
{
const auto& [params, tail] = flatten(context.arguments);
if (params.size() < 2 || params.size() > 4)
return false;
TypeArena* arena = context.solver->arena;
NotNull<BuiltinTypes> builtinTypes = context.solver->builtinTypes;
AstExprConstantString* pattern = nullptr;
size_t patternIndex = context.callSite->self ? 0 : 1;
if (context.callSite->args.size > patternIndex)
pattern = context.callSite->args.data[patternIndex]->as<AstExprConstantString>();
if (!pattern)
return false;
bool plain = false;
size_t plainIndex = context.callSite->self ? 2 : 3;
if (context.callSite->args.size > plainIndex)
{
AstExprConstantBool* p = context.callSite->args.data[plainIndex]->as<AstExprConstantBool>();
plain = p && p->value;
}
std::vector<TypeId> returnTypes;
if (!plain)
{
returnTypes = parsePatternString(builtinTypes, pattern->value.data, pattern->value.size);
if (returnTypes.empty())
return false;
}
context.solver->unify(params[0], builtinTypes->stringType, context.solver->rootScope);
const TypeId optionalNumber = arena->addType(UnionType{{builtinTypes->nilType, builtinTypes->numberType}});
const TypeId optionalBoolean = arena->addType(UnionType{{builtinTypes->nilType, builtinTypes->booleanType}});
size_t initIndex = context.callSite->self ? 1 : 2;
if (params.size() >= 3 && context.callSite->args.size > initIndex)
context.solver->unify(params[2], optionalNumber, context.solver->rootScope);
if (params.size() == 4 && context.callSite->args.size > plainIndex)
context.solver->unify(params[3], optionalBoolean, context.solver->rootScope);
returnTypes.insert(returnTypes.begin(), {optionalNumber, optionalNumber});
const TypePackId returnList = arena->addTypePack(returnTypes);
asMutable(context.result)->ty.emplace<BoundTypePack>(returnList);
return true;
}
std::vector<TypeId> filterMap(TypeId type, TypeIdPredicate predicate)
{
type = follow(type);
if (auto utv = get<UnionType>(type))
{
std::set<TypeId> options;
for (TypeId option : utv)
if (auto out = predicate(follow(option)))
options.insert(*out);
return std::vector<TypeId>(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<FunctionType>(ty))
return &ftv->tags;
else if (auto ttv = getMutable<TableType>(ty))
return &ttv->tags;
else if (auto ctv = getMutable<ClassType>(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<ClassType>(ty))
{
while (ctv)
{
if (hasTag(ctv->tags, tagName))
return true;
else if (!ctv->parent)
return false;
ctv = get<ClassType>(*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);
}
2022-08-04 23:35:33 +01:00
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