// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details #pragma once #include "Luau/Def.h" #include "Luau/LValue.h" #include "Luau/Location.h" #include "Luau/NotNull.h" #include "Luau/Type.h" #include "Luau/DenseHash.h" #include "Luau/Symbol.h" #include "Luau/Unifiable.h" #include #include #include namespace Luau { struct Scope; using ScopePtr = std::shared_ptr; struct Binding { TypeId typeId; Location location; bool deprecated = false; std::string deprecatedSuggestion; std::optional documentationSymbol; }; struct Scope { explicit Scope(TypePackId returnType); // root scope explicit Scope(const ScopePtr& parent, int subLevel = 0); // child scope. Parent must not be nullptr. const ScopePtr parent; // null for the root // All the children of this scope. std::vector> children; std::unordered_map bindings; TypePackId returnType; std::optional varargPack; TypeLevel level; Location location; // the spanning location associated with this scope std::unordered_map exportedTypeBindings; std::unordered_map privateTypeBindings; std::unordered_map typeAliasLocations; std::unordered_map typeAliasNameLocations; std::unordered_map importedModules; // Mapping from the name in the require statement to the internal moduleName. std::unordered_map> importedTypeBindings; DenseHashSet builtinTypeNames{""}; void addBuiltinTypeBinding(const Name& name, const TypeFun& tyFun); std::optional lookup(Symbol sym) const; std::optional lookupUnrefinedType(DefId def) const; std::optional lookup(DefId def) const; std::optional> lookupEx(DefId def); std::optional> lookupEx(Symbol sym); std::optional lookupType(const Name& name) const; std::optional lookupImportedType(const Name& moduleAlias, const Name& name) const; std::unordered_map privateTypePackBindings; std::optional lookupPack(const Name& name) const; // WARNING: This function linearly scans for a string key of equal value! It is thus O(n**2) std::optional linearSearchForBinding(const std::string& name, bool traverseScopeChain = true) const; RefinementMap refinements; // This can be viewed as the "unrefined" type of each binding. DenseHashMap lvalueTypes{nullptr}; // Luau values are routinely refined more narrowly than their actual // inferred type through control flow statements. We retain those refined // types here. DenseHashMap rvalueRefinements{nullptr}; void inheritAssignments(const ScopePtr& childScope); void inheritRefinements(const ScopePtr& childScope); // Track globals that should emit warnings during type checking. DenseHashSet globalsToWarn{""}; bool shouldWarnGlobal(std::string name) const; // For mutually recursive type aliases, it's important that // they use the same types for the same names. // For instance, in `type Tree { data: T, children: Forest } type Forest = {Tree}` // we need that the generic type `T` in both cases is the same, so we use a cache. std::unordered_map typeAliasTypeParameters; std::unordered_map typeAliasTypePackParameters; std::optional> interiorFreeTypes; }; // Returns true iff the left scope encloses the right scope. A Scope* equal to // nullptr is considered to be the outermost-possible scope. bool subsumesStrict(Scope* left, Scope* right); // Returns true if the left scope encloses the right scope, or if they are the // same scope. As in subsumesStrict(), nullptr is considered to be the // outermost-possible scope. bool subsumes(Scope* left, Scope* right); inline Scope* max(Scope* left, Scope* right) { if (subsumes(left, right)) return right; else return left; } } // namespace Luau