// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details #include "Luau/TypeAttach.h" #include "Luau/Error.h" #include "Luau/Module.h" #include "Luau/RecursionCounter.h" #include "Luau/Scope.h" #include "Luau/ToString.h" #include "Luau/TypeInfer.h" #include "Luau/TypePack.h" #include "Luau/TypeVar.h" #include static char* allocateString(Luau::Allocator& allocator, std::string_view contents) { char* result = (char*)allocator.allocate(contents.size() + 1); memcpy(result, contents.data(), contents.size()); result[contents.size()] = '\0'; return result; } template static char* allocateString(Luau::Allocator& allocator, const char* format, Data... data) { int len = snprintf(nullptr, 0, format, data...); char* result = (char*)allocator.allocate(len + 1); snprintf(result, len + 1, format, data...); return result; } using SyntheticNames = std::unordered_map; namespace Luau { static const char* getName(Allocator* allocator, SyntheticNames* syntheticNames, const Unifiable::Generic& gen) { size_t s = syntheticNames->size(); char*& n = (*syntheticNames)[&gen]; if (!n) { std::string str = gen.explicitName ? gen.name : generateName(s); n = static_cast(allocator->allocate(str.size() + 1)); strcpy(n, str.c_str()); } return n; } class TypeRehydrationVisitor { std::map seen; int count = 0; bool hasSeen(const void* tv) { void* ttv = const_cast(tv); auto it = seen.find(ttv); if (it != seen.end() && it->second < count) return true; seen[ttv] = count; return false; } public: TypeRehydrationVisitor(Allocator* alloc, SyntheticNames* syntheticNames, const TypeRehydrationOptions& options = TypeRehydrationOptions()) : allocator(alloc) , syntheticNames(syntheticNames) , options(options) { } AstTypePack* rehydrate(TypePackId tp); AstType* operator()(const PrimitiveTypeVar& ptv) { switch (ptv.type) { case PrimitiveTypeVar::NilType: return allocator->alloc(Location(), std::nullopt, AstName("nil")); case PrimitiveTypeVar::Boolean: return allocator->alloc(Location(), std::nullopt, AstName("boolean")); case PrimitiveTypeVar::Number: return allocator->alloc(Location(), std::nullopt, AstName("number")); case PrimitiveTypeVar::String: return allocator->alloc(Location(), std::nullopt, AstName("string")); case PrimitiveTypeVar::Thread: return allocator->alloc(Location(), std::nullopt, AstName("thread")); default: return nullptr; } } AstType* operator()(const ConstrainedTypeVar& ctv) { AstArray types; types.size = ctv.parts.size(); types.data = static_cast(allocator->allocate(sizeof(AstType*) * ctv.parts.size())); for (size_t i = 0; i < ctv.parts.size(); ++i) types.data[i] = Luau::visit(*this, ctv.parts[i]->ty); return allocator->alloc(Location(), types); } AstType* operator()(const SingletonTypeVar& stv) { if (const BooleanSingleton* bs = get(&stv)) return allocator->alloc(Location(), bs->value); else if (const StringSingleton* ss = get(&stv)) { AstArray value; value.data = const_cast(ss->value.c_str()); value.size = strlen(value.data); return allocator->alloc(Location(), value); } else return nullptr; } AstType* operator()(const AnyTypeVar&) { return allocator->alloc(Location(), std::nullopt, AstName("any")); } AstType* operator()(const TableTypeVar& ttv) { RecursionCounter counter(&count); if (ttv.name && options.bannedNames.find(*ttv.name) == options.bannedNames.end()) { AstArray parameters; parameters.size = ttv.instantiatedTypeParams.size(); parameters.data = static_cast(allocator->allocate(sizeof(AstTypeOrPack) * parameters.size)); for (size_t i = 0; i < ttv.instantiatedTypeParams.size(); ++i) { parameters.data[i] = {Luau::visit(*this, ttv.instantiatedTypeParams[i]->ty), {}}; } for (size_t i = 0; i < ttv.instantiatedTypePackParams.size(); ++i) { parameters.data[i] = {{}, rehydrate(ttv.instantiatedTypePackParams[i])}; } return allocator->alloc(Location(), std::nullopt, AstName(ttv.name->c_str()), parameters.size != 0, parameters); } if (hasSeen(&ttv)) { if (ttv.name) return allocator->alloc(Location(), std::nullopt, AstName(ttv.name->c_str())); else return allocator->alloc(Location(), std::nullopt, AstName("")); } AstArray props; props.size = ttv.props.size(); props.data = static_cast(allocator->allocate(sizeof(AstTableProp) * props.size)); int idx = 0; for (const auto& [propName, prop] : ttv.props) { RecursionCounter counter(&count); char* name = allocateString(*allocator, propName); props.data[idx].name = AstName(name); props.data[idx].type = Luau::visit(*this, prop.type->ty); props.data[idx].location = Location(); idx++; } AstTableIndexer* indexer = nullptr; if (ttv.indexer) { RecursionCounter counter(&count); indexer = allocator->alloc(); indexer->indexType = Luau::visit(*this, ttv.indexer->indexType->ty); indexer->resultType = Luau::visit(*this, ttv.indexer->indexResultType->ty); } return allocator->alloc(Location(), props, indexer); } AstType* operator()(const MetatableTypeVar& mtv) { return Luau::visit(*this, mtv.table->ty); } AstType* operator()(const ClassTypeVar& ctv) { RecursionCounter counter(&count); char* name = allocateString(*allocator, ctv.name); if (!options.expandClassProps || hasSeen(&ctv) || count > 1) return allocator->alloc(Location(), std::nullopt, AstName{name}); AstArray props; props.size = ctv.props.size(); props.data = static_cast(allocator->allocate(sizeof(AstTableProp) * props.size)); int idx = 0; for (const auto& [propName, prop] : ctv.props) { char* name = allocateString(*allocator, propName); props.data[idx].name = AstName{name}; props.data[idx].type = Luau::visit(*this, prop.type->ty); props.data[idx].location = Location(); idx++; } return allocator->alloc(Location(), props); } AstType* operator()(const FunctionTypeVar& ftv) { RecursionCounter counter(&count); if (hasSeen(&ftv)) return allocator->alloc(Location(), std::nullopt, AstName("")); AstArray generics; generics.size = ftv.generics.size(); generics.data = static_cast(allocator->allocate(sizeof(AstGenericType) * generics.size)); size_t numGenerics = 0; for (auto it = ftv.generics.begin(); it != ftv.generics.end(); ++it) { if (auto gtv = get(*it)) generics.data[numGenerics++] = {AstName(gtv->name.c_str()), Location(), nullptr}; } AstArray genericPacks; genericPacks.size = ftv.genericPacks.size(); genericPacks.data = static_cast(allocator->allocate(sizeof(AstGenericTypePack) * genericPacks.size)); size_t numGenericPacks = 0; for (auto it = ftv.genericPacks.begin(); it != ftv.genericPacks.end(); ++it) { if (auto gtv = get(*it)) genericPacks.data[numGenericPacks++] = {AstName(gtv->name.c_str()), Location(), nullptr}; } AstArray argTypes; const auto& [argVector, argTail] = flatten(ftv.argTypes); argTypes.size = argVector.size(); argTypes.data = static_cast(allocator->allocate(sizeof(AstType*) * argTypes.size)); for (size_t i = 0; i < argTypes.size; ++i) { RecursionCounter counter(&count); argTypes.data[i] = Luau::visit(*this, (argVector[i])->ty); } AstTypePack* argTailAnnotation = nullptr; if (argTail) argTailAnnotation = rehydrate(*argTail); AstArray> argNames; argNames.size = ftv.argNames.size(); argNames.data = static_cast*>(allocator->allocate(sizeof(std::optional) * argNames.size)); size_t i = 0; for (const auto& el : ftv.argNames) { std::optional* arg = &argNames.data[i++]; if (el) new (arg) std::optional(AstArgumentName(AstName(el->name.c_str()), el->location)); else new (arg) std::optional(); } AstArray returnTypes; const auto& [retVector, retTail] = flatten(ftv.retType); returnTypes.size = retVector.size(); returnTypes.data = static_cast(allocator->allocate(sizeof(AstType*) * returnTypes.size)); for (size_t i = 0; i < returnTypes.size; ++i) { RecursionCounter counter(&count); returnTypes.data[i] = Luau::visit(*this, (retVector[i])->ty); } AstTypePack* retTailAnnotation = nullptr; if (retTail) retTailAnnotation = rehydrate(*retTail); return allocator->alloc( Location(), generics, genericPacks, AstTypeList{argTypes, argTailAnnotation}, argNames, AstTypeList{returnTypes, retTailAnnotation}); } AstType* operator()(const Unifiable::Error&) { return allocator->alloc(Location(), std::nullopt, AstName("Unifiable")); } AstType* operator()(const GenericTypeVar& gtv) { return allocator->alloc(Location(), std::nullopt, AstName(getName(allocator, syntheticNames, gtv))); } AstType* operator()(const Unifiable::Bound& bound) { return Luau::visit(*this, bound.boundTo->ty); } AstType* operator()(const FreeTypeVar& ftv) { return allocator->alloc(Location(), std::nullopt, AstName("free")); } AstType* operator()(const UnionTypeVar& uv) { AstArray unionTypes; unionTypes.size = uv.options.size(); unionTypes.data = static_cast(allocator->allocate(sizeof(AstType*) * unionTypes.size)); for (size_t i = 0; i < unionTypes.size; ++i) { unionTypes.data[i] = Luau::visit(*this, uv.options[i]->ty); } return allocator->alloc(Location(), unionTypes); } AstType* operator()(const IntersectionTypeVar& uv) { AstArray intersectionTypes; intersectionTypes.size = uv.parts.size(); intersectionTypes.data = static_cast(allocator->allocate(sizeof(AstType*) * intersectionTypes.size)); for (size_t i = 0; i < intersectionTypes.size; ++i) { intersectionTypes.data[i] = Luau::visit(*this, uv.parts[i]->ty); } return allocator->alloc(Location(), intersectionTypes); } AstType* operator()(const LazyTypeVar& ltv) { return allocator->alloc(Location(), std::nullopt, AstName("")); } private: Allocator* allocator; SyntheticNames* syntheticNames; const TypeRehydrationOptions& options; }; class TypePackRehydrationVisitor { public: TypePackRehydrationVisitor(Allocator* allocator, SyntheticNames* syntheticNames, TypeRehydrationVisitor* typeVisitor) : allocator(allocator) , syntheticNames(syntheticNames) , typeVisitor(typeVisitor) { LUAU_ASSERT(allocator); LUAU_ASSERT(syntheticNames); LUAU_ASSERT(typeVisitor); } AstTypePack* operator()(const BoundTypePack& btp) const { return Luau::visit(*this, btp.boundTo->ty); } AstTypePack* operator()(const TypePack& tp) const { AstArray head; head.size = tp.head.size(); head.data = static_cast(allocator->allocate(sizeof(AstType*) * tp.head.size())); for (size_t i = 0; i < tp.head.size(); i++) head.data[i] = Luau::visit(*typeVisitor, tp.head[i]->ty); AstTypePack* tail = nullptr; if (tp.tail) tail = Luau::visit(*this, (*tp.tail)->ty); return allocator->alloc(Location(), AstTypeList{head, tail}); } AstTypePack* operator()(const VariadicTypePack& vtp) const { if (vtp.hidden) return nullptr; return allocator->alloc(Location(), Luau::visit(*typeVisitor, vtp.ty->ty)); } AstTypePack* operator()(const GenericTypePack& gtp) const { return allocator->alloc(Location(), AstName(getName(allocator, syntheticNames, gtp))); } AstTypePack* operator()(const FreeTypePack& gtp) const { return allocator->alloc(Location(), AstName("free")); } AstTypePack* operator()(const Unifiable::Error&) const { return allocator->alloc(Location(), AstName("Unifiable")); } private: Allocator* allocator; SyntheticNames* syntheticNames; TypeRehydrationVisitor* typeVisitor; }; AstTypePack* TypeRehydrationVisitor::rehydrate(TypePackId tp) { TypePackRehydrationVisitor tprv(allocator, syntheticNames, this); return Luau::visit(tprv, tp->ty); } class TypeAttacher : public AstVisitor { public: TypeAttacher(Module& checker, Luau::Allocator* alloc) : module(checker) , allocator(alloc) { } ScopePtr getScope(const Location& loc) { Location scopeLocation; ScopePtr scope = nullptr; for (const auto& s : module.scopes) { if (s.first.encloses(loc)) { if (!scope || scopeLocation.encloses(s.first)) { scopeLocation = s.first; scope = s.second; } } } return scope; } AstType* typeAst(std::optional type) { if (!type) return nullptr; return Luau::visit(TypeRehydrationVisitor(allocator, &syntheticNames), (*type)->ty); } AstArray typeAstPack(TypePackId type) { const auto& [v, tail] = flatten(type); AstArray result; result.size = v.size(); result.data = static_cast(allocator->allocate(sizeof(AstType*) * v.size())); for (size_t i = 0; i < v.size(); ++i) { result.data[i] = Luau::visit(TypeRehydrationVisitor(allocator, &syntheticNames), v[i]->ty); } return result; } virtual bool visit(AstStatLocal* al) override { for (size_t i = 0; i < al->vars.size; ++i) { visitLocal(al->vars.data[i]); } return true; } virtual bool visitLocal(AstLocal* local) { AstType* annotation = local->annotation; if (!annotation) { if (auto result = getScope(local->location)->lookup(local)) local->annotation = typeAst(*result); } return true; } virtual bool visit(AstExprLocal* al) override { return visitLocal(al->local); } virtual bool visit(AstStatFor* stat) override { visitLocal(stat->var); return true; } virtual bool visit(AstStatForIn* stat) override { for (size_t i = 0; i < stat->vars.size; ++i) visitLocal(stat->vars.data[i]); return true; } virtual bool visit(AstExprFunction* fn) override { // TODO: add generics if the inferred type of the function is generic CLI-39908 for (size_t i = 0; i < fn->args.size; ++i) { AstLocal* arg = fn->args.data[i]; visitLocal(arg); } if (!fn->returnAnnotation) { if (auto result = getScope(fn->body->location)) { TypePackId ret = result->returnType; AstTypePack* variadicAnnotation = nullptr; const auto& [v, tail] = flatten(ret); if (tail) variadicAnnotation = TypeRehydrationVisitor(allocator, &syntheticNames).rehydrate(*tail); fn->returnAnnotation = AstTypeList{typeAstPack(ret), variadicAnnotation}; } } return true; } private: Module& module; Allocator* allocator; SyntheticNames syntheticNames; }; void attachTypeData(SourceModule& source, Module& result) { TypeAttacher ta(result, source.allocator.get()); source.root->visit(&ta); } AstType* rehydrateAnnotation(TypeId type, Allocator* allocator, const TypeRehydrationOptions& options) { SyntheticNames syntheticNames; return Luau::visit(TypeRehydrationVisitor(allocator, &syntheticNames, options), type->ty); } } // namespace Luau