// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details #include "Luau/ConstraintGraphBuilder.h" #include "Luau/Ast.h" #include "Luau/Clone.h" #include "Luau/Common.h" #include "Luau/Constraint.h" #include "Luau/DcrLogger.h" #include "Luau/ModuleResolver.h" #include "Luau/RecursionCounter.h" #include "Luau/Scope.h" #include "Luau/ToString.h" #include "Luau/TypeUtils.h" #include "Luau/TypeVar.h" LUAU_FASTINT(LuauCheckRecursionLimit); LUAU_FASTFLAG(DebugLuauLogSolverToJson); LUAU_FASTFLAG(DebugLuauMagicTypes); namespace Luau { const AstStat* getFallthrough(const AstStat* node); // TypeInfer.cpp static std::optional matchRequire(const AstExprCall& call) { const char* require = "require"; if (call.args.size != 1) return std::nullopt; const AstExprGlobal* funcAsGlobal = call.func->as(); if (!funcAsGlobal || funcAsGlobal->name != require) return std::nullopt; if (call.args.size != 1) return std::nullopt; return call.args.data[0]; } static bool matchSetmetatable(const AstExprCall& call) { const char* smt = "setmetatable"; if (call.args.size != 2) return false; const AstExprGlobal* funcAsGlobal = call.func->as(); if (!funcAsGlobal || funcAsGlobal->name != smt) return false; return true; } struct TypeGuard { bool isTypeof; AstExpr* target; std::string type; }; static std::optional matchTypeGuard(const AstExprBinary* binary) { if (binary->op != AstExprBinary::CompareEq && binary->op != AstExprBinary::CompareNe) return std::nullopt; AstExpr* left = binary->left; AstExpr* right = binary->right; if (right->is()) std::swap(left, right); if (!right->is()) return std::nullopt; AstExprCall* call = left->as(); AstExprConstantString* string = right->as(); if (!call || !string) return std::nullopt; AstExprGlobal* callee = call->func->as(); if (!callee) return std::nullopt; if (callee->name != "type" && callee->name != "typeof") return std::nullopt; if (call->args.size != 1) return std::nullopt; return TypeGuard{ /*isTypeof*/ callee->name == "typeof", /*target*/ call->args.data[0], /*type*/ std::string(string->value.data, string->value.size), }; } namespace { struct Checkpoint { size_t offset; }; Checkpoint checkpoint(const ConstraintGraphBuilder* cgb) { return Checkpoint{cgb->constraints.size()}; } template void forEachConstraint(const Checkpoint& start, const Checkpoint& end, const ConstraintGraphBuilder* cgb, F f) { for (size_t i = start.offset; i < end.offset; ++i) f(cgb->constraints[i]); } } // namespace ConstraintGraphBuilder::ConstraintGraphBuilder(const ModuleName& moduleName, ModulePtr module, TypeArena* arena, NotNull moduleResolver, NotNull singletonTypes, NotNull ice, const ScopePtr& globalScope, DcrLogger* logger, NotNull dfg) : moduleName(moduleName) , module(module) , singletonTypes(singletonTypes) , arena(arena) , rootScope(nullptr) , dfg(dfg) , moduleResolver(moduleResolver) , ice(ice) , globalScope(globalScope) , logger(logger) { if (FFlag::DebugLuauLogSolverToJson) LUAU_ASSERT(logger); LUAU_ASSERT(module); } TypeId ConstraintGraphBuilder::freshType(const ScopePtr& scope) { return arena->addType(FreeTypeVar{scope.get()}); } TypePackId ConstraintGraphBuilder::freshTypePack(const ScopePtr& scope) { FreeTypePack f{scope.get()}; return arena->addTypePack(TypePackVar{std::move(f)}); } ScopePtr ConstraintGraphBuilder::childScope(AstNode* node, const ScopePtr& parent) { auto scope = std::make_shared(parent); scopes.emplace_back(node->location, scope); scope->returnType = parent->returnType; scope->varargPack = parent->varargPack; parent->children.push_back(NotNull{scope.get()}); module->astScopes[node] = scope.get(); return scope; } NotNull ConstraintGraphBuilder::addConstraint(const ScopePtr& scope, const Location& location, ConstraintV cv) { return NotNull{constraints.emplace_back(new Constraint{NotNull{scope.get()}, location, std::move(cv)}).get()}; } NotNull ConstraintGraphBuilder::addConstraint(const ScopePtr& scope, std::unique_ptr c) { return NotNull{constraints.emplace_back(std::move(c)).get()}; } static void unionRefinements(const std::unordered_map& lhs, const std::unordered_map& rhs, std::unordered_map& dest, NotNull arena) { for (auto [def, ty] : lhs) { auto rhsIt = rhs.find(def); if (rhsIt == rhs.end()) continue; std::vector discriminants{{ty, rhsIt->second}}; if (auto destIt = dest.find(def); destIt != dest.end()) discriminants.push_back(destIt->second); dest[def] = arena->addType(UnionTypeVar{std::move(discriminants)}); } } static void computeRefinement(const ScopePtr& scope, ConnectiveId connective, std::unordered_map* refis, bool sense, NotNull arena, bool eq, std::vector* constraints) { using RefinementMap = std::unordered_map; if (!connective) return; else if (auto negation = get(connective)) return computeRefinement(scope, negation->connective, refis, !sense, arena, eq, constraints); else if (auto conjunction = get(connective)) { RefinementMap lhsRefis; RefinementMap rhsRefis; computeRefinement(scope, conjunction->lhs, sense ? refis : &lhsRefis, sense, arena, eq, constraints); computeRefinement(scope, conjunction->rhs, sense ? refis : &rhsRefis, sense, arena, eq, constraints); if (!sense) unionRefinements(lhsRefis, rhsRefis, *refis, arena); } else if (auto disjunction = get(connective)) { RefinementMap lhsRefis; RefinementMap rhsRefis; computeRefinement(scope, disjunction->lhs, sense ? &lhsRefis : refis, sense, arena, eq, constraints); computeRefinement(scope, disjunction->rhs, sense ? &rhsRefis : refis, sense, arena, eq, constraints); if (sense) unionRefinements(lhsRefis, rhsRefis, *refis, arena); } else if (auto equivalence = get(connective)) { computeRefinement(scope, equivalence->lhs, refis, sense, arena, true, constraints); computeRefinement(scope, equivalence->rhs, refis, sense, arena, true, constraints); } else if (auto proposition = get(connective)) { TypeId discriminantTy = proposition->discriminantTy; if (!sense && !eq) discriminantTy = arena->addType(NegationTypeVar{proposition->discriminantTy}); else if (!sense && eq) { discriminantTy = arena->addType(BlockedTypeVar{}); constraints->push_back(SingletonOrTopTypeConstraint{discriminantTy, proposition->discriminantTy}); } if (auto it = refis->find(proposition->def); it != refis->end()) (*refis)[proposition->def] = arena->addType(IntersectionTypeVar{{discriminantTy, it->second}}); else (*refis)[proposition->def] = discriminantTy; } } void ConstraintGraphBuilder::applyRefinements(const ScopePtr& scope, Location location, ConnectiveId connective) { if (!connective) return; std::unordered_map refinements; std::vector constraints; computeRefinement(scope, connective, &refinements, /*sense*/ true, arena, /*eq*/ false, &constraints); for (auto [def, discriminantTy] : refinements) { std::optional defTy = scope->lookup(def); if (!defTy) ice->ice("Every DefId must map to a type!"); TypeId resultTy = arena->addType(IntersectionTypeVar{{*defTy, discriminantTy}}); scope->dcrRefinements[def] = resultTy; } for (auto& c : constraints) addConstraint(scope, location, c); } void ConstraintGraphBuilder::visit(AstStatBlock* block) { LUAU_ASSERT(scopes.empty()); LUAU_ASSERT(rootScope == nullptr); ScopePtr scope = std::make_shared(globalScope); rootScope = scope.get(); scopes.emplace_back(block->location, scope); module->astScopes[block] = NotNull{scope.get()}; rootScope->returnType = freshTypePack(scope); prepopulateGlobalScope(scope, block); visitBlockWithoutChildScope(scope, block); } void ConstraintGraphBuilder::visitBlockWithoutChildScope(const ScopePtr& scope, AstStatBlock* block) { RecursionCounter counter{&recursionCount}; if (recursionCount >= FInt::LuauCheckRecursionLimit) { reportCodeTooComplex(block->location); return; } std::unordered_map aliasDefinitionLocations; // In order to enable mutually-recursive type aliases, we need to // populate the type bindings before we actually check any of the // alias statements. Since we're not ready to actually resolve // any of the annotations, we just use a fresh type for now. for (AstStat* stat : block->body) { if (auto alias = stat->as()) { if (scope->privateTypeBindings.count(alias->name.value) != 0) { auto it = aliasDefinitionLocations.find(alias->name.value); LUAU_ASSERT(it != aliasDefinitionLocations.end()); reportError(alias->location, DuplicateTypeDefinition{alias->name.value, it->second}); continue; } bool hasGenerics = alias->generics.size > 0 || alias->genericPacks.size > 0; ScopePtr defnScope = scope; if (hasGenerics) { defnScope = childScope(alias, scope); } TypeId initialType = freshType(scope); TypeFun initialFun = TypeFun{initialType}; for (const auto& [name, gen] : createGenerics(defnScope, alias->generics)) { initialFun.typeParams.push_back(gen); defnScope->privateTypeBindings[name] = TypeFun{gen.ty}; } for (const auto& [name, genPack] : createGenericPacks(defnScope, alias->genericPacks)) { initialFun.typePackParams.push_back(genPack); defnScope->privateTypePackBindings[name] = genPack.tp; } scope->privateTypeBindings[alias->name.value] = std::move(initialFun); astTypeAliasDefiningScopes[alias] = defnScope; aliasDefinitionLocations[alias->name.value] = alias->location; } } for (AstStat* stat : block->body) visit(scope, stat); } void ConstraintGraphBuilder::visit(const ScopePtr& scope, AstStat* stat) { RecursionLimiter limiter{&recursionCount, FInt::LuauCheckRecursionLimit}; if (auto s = stat->as()) visit(scope, s); else if (auto s = stat->as()) visit(scope, s); else if (auto s = stat->as()) visit(scope, s); else if (auto s = stat->as()) visit(scope, s); else if (auto s = stat->as()) visit(scope, s); else if (auto s = stat->as()) visit(scope, s); else if (auto f = stat->as()) visit(scope, f); else if (auto f = stat->as()) visit(scope, f); else if (auto r = stat->as()) visit(scope, r); else if (auto a = stat->as()) visit(scope, a); else if (auto a = stat->as()) visit(scope, a); else if (auto e = stat->as()) checkPack(scope, e->expr); else if (auto i = stat->as()) visit(scope, i); else if (auto a = stat->as()) visit(scope, a); else if (auto s = stat->as()) visit(scope, s); else if (auto s = stat->as()) visit(scope, s); else if (auto s = stat->as()) visit(scope, s); else if (auto s = stat->as()) visit(scope, s); else LUAU_ASSERT(0); } void ConstraintGraphBuilder::visit(const ScopePtr& scope, AstStatLocal* local) { std::vector varTypes; varTypes.reserve(local->vars.size); for (AstLocal* local : local->vars) { TypeId ty = nullptr; if (local->annotation) ty = resolveType(scope, local->annotation, /* topLevel */ true); varTypes.push_back(ty); } for (size_t i = 0; i < local->values.size; ++i) { AstExpr* value = local->values.data[i]; const bool hasAnnotation = i < local->vars.size && nullptr != local->vars.data[i]->annotation; if (value->is()) { // HACK: we leave nil-initialized things floating under the // assumption that they will later be populated. // // See the test TypeInfer/infer_locals_with_nil_value. Better flow // awareness should make this obsolete. if (!varTypes[i]) varTypes[i] = freshType(scope); } // Only function calls and vararg expressions can produce packs. All // other expressions produce exactly one value. else if (i != local->values.size - 1 || (!value->is() && !value->is())) { std::optional expectedType; if (hasAnnotation) expectedType = varTypes.at(i); TypeId exprType = check(scope, value, expectedType).ty; if (i < varTypes.size()) { if (varTypes[i]) addConstraint(scope, local->location, SubtypeConstraint{exprType, varTypes[i]}); else varTypes[i] = exprType; } } else { std::vector expectedTypes; if (hasAnnotation) expectedTypes.insert(begin(expectedTypes), begin(varTypes) + i, end(varTypes)); TypePackId exprPack = checkPack(scope, value, expectedTypes).tp; if (i < local->vars.size) { std::vector packTypes = flatten(*arena, singletonTypes, exprPack, varTypes.size() - i); // fill out missing values in varTypes with values from exprPack for (size_t j = i; j < varTypes.size(); ++j) { if (!varTypes[j]) { if (j - i < packTypes.size()) varTypes[j] = packTypes[j - i]; else varTypes[j] = freshType(scope); } } std::vector tailValues{varTypes.begin() + i, varTypes.end()}; TypePackId tailPack = arena->addTypePack(std::move(tailValues)); addConstraint(scope, local->location, PackSubtypeConstraint{exprPack, tailPack}); } } } for (size_t i = 0; i < local->vars.size; ++i) { AstLocal* l = local->vars.data[i]; Location location = l->location; if (!varTypes[i]) varTypes[i] = freshType(scope); scope->bindings[l] = Binding{varTypes[i], location}; // HACK: In the greedy solver, we say the type state of a variable is the type annotation itself, but // the actual type state is the corresponding initializer expression (if it exists) or nil otherwise. if (auto def = dfg->getDef(l)) scope->dcrRefinements[*def] = varTypes[i]; } if (local->values.size > 0) { // To correctly handle 'require', we need to import the exported type bindings into the variable 'namespace'. for (size_t i = 0; i < local->values.size && i < local->vars.size; ++i) { const AstExprCall* call = local->values.data[i]->as(); if (!call) continue; if (auto maybeRequire = matchRequire(*call)) { AstExpr* require = *maybeRequire; if (auto moduleInfo = moduleResolver->resolveModuleInfo(moduleName, *require)) { const Name name{local->vars.data[i]->name.value}; if (ModulePtr module = moduleResolver->getModule(moduleInfo->name)) scope->importedTypeBindings[name] = module->getModuleScope()->exportedTypeBindings; } } } } } void ConstraintGraphBuilder::visit(const ScopePtr& scope, AstStatFor* for_) { auto checkNumber = [&](AstExpr* expr) { if (!expr) return; TypeId t = check(scope, expr).ty; addConstraint(scope, expr->location, SubtypeConstraint{t, singletonTypes->numberType}); }; checkNumber(for_->from); checkNumber(for_->to); checkNumber(for_->step); ScopePtr forScope = childScope(for_, scope); forScope->bindings[for_->var] = Binding{singletonTypes->numberType, for_->var->location}; visit(forScope, for_->body); } void ConstraintGraphBuilder::visit(const ScopePtr& scope, AstStatForIn* forIn) { ScopePtr loopScope = childScope(forIn, scope); TypePackId iterator = checkPack(scope, forIn->values).tp; std::vector variableTypes; variableTypes.reserve(forIn->vars.size); for (AstLocal* var : forIn->vars) { TypeId ty = freshType(loopScope); loopScope->bindings[var] = Binding{ty, var->location}; variableTypes.push_back(ty); if (auto def = dfg->getDef(var)) loopScope->dcrRefinements[*def] = ty; } // It is always ok to provide too few variables, so we give this pack a free tail. TypePackId variablePack = arena->addTypePack(std::move(variableTypes), arena->addTypePack(FreeTypePack{loopScope.get()})); addConstraint(loopScope, getLocation(forIn->values), IterableConstraint{iterator, variablePack}); visit(loopScope, forIn->body); } void ConstraintGraphBuilder::visit(const ScopePtr& scope, AstStatWhile* while_) { check(scope, while_->condition); ScopePtr whileScope = childScope(while_, scope); visit(whileScope, while_->body); } void ConstraintGraphBuilder::visit(const ScopePtr& scope, AstStatRepeat* repeat) { ScopePtr repeatScope = childScope(repeat, scope); visit(repeatScope, repeat->body); // The condition does indeed have access to bindings from within the body of // the loop. check(repeatScope, repeat->condition); } void ConstraintGraphBuilder::visit(const ScopePtr& scope, AstStatLocalFunction* function) { // Local // Global // Dotted path // Self? TypeId functionType = nullptr; auto ty = scope->lookup(function->name); LUAU_ASSERT(!ty.has_value()); // The parser ensures that every local function has a distinct Symbol for its name. functionType = arena->addType(BlockedTypeVar{}); scope->bindings[function->name] = Binding{functionType, function->name->location}; FunctionSignature sig = checkFunctionSignature(scope, function->func); sig.bodyScope->bindings[function->name] = Binding{sig.signature, function->func->location}; auto start = checkpoint(this); checkFunctionBody(sig.bodyScope, function->func); auto end = checkpoint(this); NotNull constraintScope{sig.signatureScope ? sig.signatureScope.get() : sig.bodyScope.get()}; std::unique_ptr c = std::make_unique(constraintScope, function->name->location, GeneralizationConstraint{functionType, sig.signature}); forEachConstraint(start, end, this, [&c](const ConstraintPtr& constraint) { c->dependencies.push_back(NotNull{constraint.get()}); }); addConstraint(scope, std::move(c)); } void ConstraintGraphBuilder::visit(const ScopePtr& scope, AstStatFunction* function) { // Name could be AstStatLocal, AstStatGlobal, AstStatIndexName. // With or without self TypeId functionType = nullptr; FunctionSignature sig = checkFunctionSignature(scope, function->func); if (AstExprLocal* localName = function->name->as()) { std::optional existingFunctionTy = scope->lookup(localName->local); if (existingFunctionTy) { // Duplicate definition functionType = *existingFunctionTy; } else { functionType = arena->addType(BlockedTypeVar{}); scope->bindings[localName->local] = Binding{functionType, localName->location}; } sig.bodyScope->bindings[localName->local] = Binding{sig.signature, localName->location}; } else if (AstExprGlobal* globalName = function->name->as()) { std::optional existingFunctionTy = scope->lookup(globalName->name); if (existingFunctionTy) { // Duplicate definition functionType = *existingFunctionTy; } else { functionType = arena->addType(BlockedTypeVar{}); rootScope->bindings[globalName->name] = Binding{functionType, globalName->location}; } sig.bodyScope->bindings[globalName->name] = Binding{sig.signature, globalName->location}; } else if (AstExprIndexName* indexName = function->name->as()) { TypeId containingTableType = check(scope, indexName->expr).ty; functionType = arena->addType(BlockedTypeVar{}); // TODO look into stack utilization. This is probably ok because it scales with AST depth. TypeId prospectiveTableType = arena->addType(TableTypeVar{TableState::Unsealed, TypeLevel{}, scope.get()}); NotNull prospectiveTable{getMutable(prospectiveTableType)}; Property& prop = prospectiveTable->props[indexName->index.value]; prop.type = functionType; prop.location = function->name->location; addConstraint(scope, indexName->location, SubtypeConstraint{containingTableType, prospectiveTableType}); } else if (AstExprError* err = function->name->as()) { functionType = singletonTypes->errorRecoveryType(); } LUAU_ASSERT(functionType != nullptr); auto start = checkpoint(this); checkFunctionBody(sig.bodyScope, function->func); auto end = checkpoint(this); NotNull constraintScope{sig.signatureScope ? sig.signatureScope.get() : sig.bodyScope.get()}; std::unique_ptr c = std::make_unique(constraintScope, function->name->location, GeneralizationConstraint{functionType, sig.signature}); forEachConstraint(start, end, this, [&c](const ConstraintPtr& constraint) { c->dependencies.push_back(NotNull{constraint.get()}); }); addConstraint(scope, std::move(c)); } void ConstraintGraphBuilder::visit(const ScopePtr& scope, AstStatReturn* ret) { // At this point, the only way scope->returnType should have anything // interesting in it is if the function has an explicit return annotation. // If this is the case, then we can expect that the return expression // conforms to that. std::vector expectedTypes; for (TypeId ty : scope->returnType) expectedTypes.push_back(ty); TypePackId exprTypes = checkPack(scope, ret->list, expectedTypes).tp; addConstraint(scope, ret->location, PackSubtypeConstraint{exprTypes, scope->returnType}); } void ConstraintGraphBuilder::visit(const ScopePtr& scope, AstStatBlock* block) { ScopePtr innerScope = childScope(block, scope); visitBlockWithoutChildScope(innerScope, block); } void ConstraintGraphBuilder::visit(const ScopePtr& scope, AstStatAssign* assign) { TypePackId varPackId = checkLValues(scope, assign->vars); TypePackId valuePack = checkPack(scope, assign->values).tp; addConstraint(scope, assign->location, PackSubtypeConstraint{valuePack, varPackId}); } void ConstraintGraphBuilder::visit(const ScopePtr& scope, AstStatCompoundAssign* assign) { // Synthesize A = A op B from A op= B and then build constraints for that instead. AstExprBinary exprBinary{assign->location, assign->op, assign->var, assign->value}; AstExpr* exprBinaryPtr = &exprBinary; AstArray vars{&assign->var, 1}; AstArray values{&exprBinaryPtr, 1}; AstStatAssign syntheticAssign{assign->location, vars, values}; visit(scope, &syntheticAssign); } void ConstraintGraphBuilder::visit(const ScopePtr& scope, AstStatIf* ifStatement) { // TODO: Optimization opportunity, the interior scope of the condition could be // reused for the then body, so we don't need to refine twice. ScopePtr condScope = childScope(ifStatement->condition, scope); auto [_, connective] = check(condScope, ifStatement->condition, std::nullopt); ScopePtr thenScope = childScope(ifStatement->thenbody, scope); applyRefinements(thenScope, Location{}, connective); visit(thenScope, ifStatement->thenbody); if (ifStatement->elsebody) { ScopePtr elseScope = childScope(ifStatement->elsebody, scope); applyRefinements(elseScope, Location{}, connectiveArena.negation(connective)); visit(elseScope, ifStatement->elsebody); } } void ConstraintGraphBuilder::visit(const ScopePtr& scope, AstStatTypeAlias* alias) { auto bindingIt = scope->privateTypeBindings.find(alias->name.value); ScopePtr* defnIt = astTypeAliasDefiningScopes.find(alias); // These will be undefined if the alias was a duplicate definition, in which // case we just skip over it. if (bindingIt == scope->privateTypeBindings.end() || defnIt == nullptr) { return; } ScopePtr resolvingScope = *defnIt; TypeId ty = resolveType(resolvingScope, alias->type, /* topLevel */ true); if (alias->exported) { Name typeName(alias->name.value); scope->exportedTypeBindings[typeName] = TypeFun{ty}; } LUAU_ASSERT(get(bindingIt->second.type)); // Rather than using a subtype constraint, we instead directly bind // the free type we generated in the first pass to the resolved type. // This prevents a case where you could cause another constraint to // bind the free alias type to an unrelated type, causing havoc. asMutable(bindingIt->second.type)->ty.emplace(ty); addConstraint(scope, alias->location, NameConstraint{ty, alias->name.value}); } void ConstraintGraphBuilder::visit(const ScopePtr& scope, AstStatDeclareGlobal* global) { LUAU_ASSERT(global->type); TypeId globalTy = resolveType(scope, global->type); Name globalName(global->name.value); module->declaredGlobals[globalName] = globalTy; scope->bindings[global->name] = Binding{globalTy, global->location}; } static bool isMetamethod(const Name& name) { return name == "__index" || name == "__newindex" || name == "__call" || name == "__concat" || name == "__unm" || name == "__add" || name == "__sub" || name == "__mul" || name == "__div" || name == "__mod" || name == "__pow" || name == "__tostring" || name == "__metatable" || name == "__eq" || name == "__lt" || name == "__le" || name == "__mode" || name == "__iter" || name == "__len"; } void ConstraintGraphBuilder::visit(const ScopePtr& scope, AstStatDeclareClass* declaredClass) { std::optional superTy = std::nullopt; if (declaredClass->superName) { Name superName = Name(declaredClass->superName->value); std::optional lookupType = scope->lookupType(superName); if (!lookupType) { reportError(declaredClass->location, UnknownSymbol{superName, UnknownSymbol::Type}); return; } // We don't have generic classes, so this assertion _should_ never be hit. LUAU_ASSERT(lookupType->typeParams.size() == 0 && lookupType->typePackParams.size() == 0); superTy = lookupType->type; if (!get(follow(*superTy))) { reportError(declaredClass->location, GenericError{format("Cannot use non-class type '%s' as a superclass of class '%s'", superName.c_str(), declaredClass->name.value)}); return; } } Name className(declaredClass->name.value); TypeId classTy = arena->addType(ClassTypeVar(className, {}, superTy, std::nullopt, {}, {}, moduleName)); ClassTypeVar* ctv = getMutable(classTy); TypeId metaTy = arena->addType(TableTypeVar{TableState::Sealed, scope->level, scope.get()}); TableTypeVar* metatable = getMutable(metaTy); ctv->metatable = metaTy; scope->exportedTypeBindings[className] = TypeFun{{}, classTy}; for (const AstDeclaredClassProp& prop : declaredClass->props) { Name propName(prop.name.value); TypeId propTy = resolveType(scope, prop.ty); bool assignToMetatable = isMetamethod(propName); // Function types always take 'self', but this isn't reflected in the // parsed annotation. Add it here. if (prop.isMethod) { if (FunctionTypeVar* ftv = getMutable(propTy)) { ftv->argNames.insert(ftv->argNames.begin(), FunctionArgument{"self", {}}); ftv->argTypes = arena->addTypePack(TypePack{{classTy}, ftv->argTypes}); ftv->hasSelf = true; } } if (ctv->props.count(propName) == 0) { if (assignToMetatable) metatable->props[propName] = {propTy}; else ctv->props[propName] = {propTy}; } else { TypeId currentTy = assignToMetatable ? metatable->props[propName].type : ctv->props[propName].type; // We special-case this logic to keep the intersection flat; otherwise we // would create a ton of nested intersection types. if (const IntersectionTypeVar* itv = get(currentTy)) { std::vector options = itv->parts; options.push_back(propTy); TypeId newItv = arena->addType(IntersectionTypeVar{std::move(options)}); if (assignToMetatable) metatable->props[propName] = {newItv}; else ctv->props[propName] = {newItv}; } else if (get(currentTy)) { TypeId intersection = arena->addType(IntersectionTypeVar{{currentTy, propTy}}); if (assignToMetatable) metatable->props[propName] = {intersection}; else ctv->props[propName] = {intersection}; } else { reportError(declaredClass->location, GenericError{format("Cannot overload non-function class member '%s'", propName.c_str())}); } } } } void ConstraintGraphBuilder::visit(const ScopePtr& scope, AstStatDeclareFunction* global) { std::vector> generics = createGenerics(scope, global->generics); std::vector> genericPacks = createGenericPacks(scope, global->genericPacks); std::vector genericTys; genericTys.reserve(generics.size()); for (auto& [name, generic] : generics) { genericTys.push_back(generic.ty); scope->privateTypeBindings[name] = TypeFun{generic.ty}; } std::vector genericTps; genericTps.reserve(genericPacks.size()); for (auto& [name, generic] : genericPacks) { genericTps.push_back(generic.tp); scope->privateTypePackBindings[name] = generic.tp; } ScopePtr funScope = scope; if (!generics.empty() || !genericPacks.empty()) funScope = childScope(global, scope); TypePackId paramPack = resolveTypePack(funScope, global->params); TypePackId retPack = resolveTypePack(funScope, global->retTypes); TypeId fnType = arena->addType(FunctionTypeVar{TypeLevel{}, funScope.get(), std::move(genericTys), std::move(genericTps), paramPack, retPack}); FunctionTypeVar* ftv = getMutable(fnType); ftv->argNames.reserve(global->paramNames.size); for (const auto& el : global->paramNames) ftv->argNames.push_back(FunctionArgument{el.first.value, el.second}); Name fnName(global->name.value); module->declaredGlobals[fnName] = fnType; scope->bindings[global->name] = Binding{fnType, global->location}; } void ConstraintGraphBuilder::visit(const ScopePtr& scope, AstStatError* error) { for (AstStat* stat : error->statements) visit(scope, stat); for (AstExpr* expr : error->expressions) check(scope, expr); } InferencePack ConstraintGraphBuilder::checkPack(const ScopePtr& scope, AstArray exprs, const std::vector& expectedTypes) { std::vector head; std::optional tail; for (size_t i = 0; i < exprs.size; ++i) { AstExpr* expr = exprs.data[i]; if (i < exprs.size - 1) { std::optional expectedType; if (i < expectedTypes.size()) expectedType = expectedTypes[i]; head.push_back(check(scope, expr).ty); } else { std::vector expectedTailTypes; if (i < expectedTypes.size()) expectedTailTypes.assign(begin(expectedTypes) + i, end(expectedTypes)); tail = checkPack(scope, expr, expectedTailTypes).tp; } } if (head.empty() && tail) return InferencePack{*tail}; else return InferencePack{arena->addTypePack(TypePack{std::move(head), tail})}; } InferencePack ConstraintGraphBuilder::checkPack(const ScopePtr& scope, AstExpr* expr, const std::vector& expectedTypes) { RecursionCounter counter{&recursionCount}; if (recursionCount >= FInt::LuauCheckRecursionLimit) { reportCodeTooComplex(expr->location); return InferencePack{singletonTypes->errorRecoveryTypePack()}; } InferencePack result; if (AstExprCall* call = expr->as()) result = {checkPack(scope, call, expectedTypes)}; else if (AstExprVarargs* varargs = expr->as()) { if (scope->varargPack) result = InferencePack{*scope->varargPack}; else result = InferencePack{singletonTypes->errorRecoveryTypePack()}; } else { std::optional expectedType; if (!expectedTypes.empty()) expectedType = expectedTypes[0]; TypeId t = check(scope, expr, expectedType).ty; result = InferencePack{arena->addTypePack({t})}; } LUAU_ASSERT(result.tp); astTypePacks[expr] = result.tp; return result; } InferencePack ConstraintGraphBuilder::checkPack(const ScopePtr& scope, AstExprCall* call, const std::vector& expectedTypes) { TypeId fnType = check(scope, call->func).ty; auto startCheckpoint = checkpoint(this); std::vector args; for (AstExpr* arg : call->args) { args.push_back(check(scope, arg).ty); } if (call->self) { AstExprIndexName* indexExpr = call->func->as(); if (!indexExpr) ice->ice("method call expression has no 'self'"); // The call to `check` we already did on `call->func` should have already produced a type for // `indexExpr->expr`, so we can get it from `astTypes` to avoid exponential blow-up. TypeId selfType = astTypes[indexExpr->expr]; // If we don't have a type for self, it means we had a code too complex error already. if (selfType == nullptr) selfType = singletonTypes->errorRecoveryType(); args.insert(args.begin(), selfType); } if (matchSetmetatable(*call)) { LUAU_ASSERT(args.size() == 2); TypeId target = args[0]; TypeId mt = args[1]; AstExpr* targetExpr = call->args.data[0]; MetatableTypeVar mtv{target, mt}; TypeId resultTy = arena->addType(mtv); if (AstExprLocal* targetLocal = targetExpr->as()) scope->bindings[targetLocal->local].typeId = resultTy; return InferencePack{arena->addTypePack({resultTy})}; } else { auto endCheckpoint = checkpoint(this); astOriginalCallTypes[call->func] = fnType; TypeId instantiatedType = arena->addType(BlockedTypeVar{}); // TODO: How do expectedTypes play into this? Do they? TypePackId rets = arena->addTypePack(BlockedTypePack{}); TypePackId argPack = arena->addTypePack(TypePack{args, {}}); FunctionTypeVar ftv(TypeLevel{}, scope.get(), argPack, rets); TypeId inferredFnType = arena->addType(ftv); unqueuedConstraints.push_back( std::make_unique(NotNull{scope.get()}, call->func->location, InstantiationConstraint{instantiatedType, fnType})); NotNull ic(unqueuedConstraints.back().get()); unqueuedConstraints.push_back( std::make_unique(NotNull{scope.get()}, call->func->location, SubtypeConstraint{inferredFnType, instantiatedType})); NotNull sc(unqueuedConstraints.back().get()); // We force constraints produced by checking function arguments to wait // until after we have resolved the constraint on the function itself. // This ensures, for instance, that we start inferring the contents of // lambdas under the assumption that their arguments and return types // will be compatible with the enclosing function call. forEachConstraint(startCheckpoint, endCheckpoint, this, [sc](const ConstraintPtr& constraint) { constraint->dependencies.push_back(sc); }); addConstraint(scope, call->func->location, FunctionCallConstraint{ {ic, sc}, fnType, argPack, rets, call, }); return InferencePack{rets}; } } Inference ConstraintGraphBuilder::check(const ScopePtr& scope, AstExpr* expr, std::optional expectedType, bool forceSingleton) { RecursionCounter counter{&recursionCount}; if (recursionCount >= FInt::LuauCheckRecursionLimit) { reportCodeTooComplex(expr->location); return Inference{singletonTypes->errorRecoveryType()}; } Inference result; if (auto group = expr->as()) result = check(scope, group->expr, expectedType, forceSingleton); else if (auto stringExpr = expr->as()) result = check(scope, stringExpr, expectedType, forceSingleton); else if (expr->is()) result = Inference{singletonTypes->numberType}; else if (auto boolExpr = expr->as()) result = check(scope, boolExpr, expectedType, forceSingleton); else if (expr->is()) result = Inference{singletonTypes->nilType}; else if (auto local = expr->as()) result = check(scope, local); else if (auto global = expr->as()) result = check(scope, global); else if (expr->is()) result = flattenPack(scope, expr->location, checkPack(scope, expr)); else if (auto call = expr->as()) { std::vector expectedTypes; if (expectedType) expectedTypes.push_back(*expectedType); result = flattenPack(scope, expr->location, checkPack(scope, call, expectedTypes)); // TODO: needs predicates too } else if (auto a = expr->as()) { FunctionSignature sig = checkFunctionSignature(scope, a); checkFunctionBody(sig.bodyScope, a); return Inference{sig.signature}; } else if (auto indexName = expr->as()) result = check(scope, indexName); else if (auto indexExpr = expr->as()) result = check(scope, indexExpr); else if (auto table = expr->as()) result = check(scope, table, expectedType); else if (auto unary = expr->as()) result = check(scope, unary); else if (auto binary = expr->as()) result = check(scope, binary, expectedType); else if (auto ifElse = expr->as()) result = check(scope, ifElse, expectedType); else if (auto typeAssert = expr->as()) result = check(scope, typeAssert); else if (auto err = expr->as()) { // Open question: Should we traverse into this? for (AstExpr* subExpr : err->expressions) check(scope, subExpr); result = Inference{singletonTypes->errorRecoveryType()}; } else { LUAU_ASSERT(0); result = Inference{freshType(scope)}; } LUAU_ASSERT(result.ty); astTypes[expr] = result.ty; return result; } Inference ConstraintGraphBuilder::check(const ScopePtr& scope, AstExprConstantString* string, std::optional expectedType, bool forceSingleton) { if (forceSingleton) return Inference{arena->addType(SingletonTypeVar{StringSingleton{std::string{string->value.data, string->value.size}}})}; if (expectedType) { const TypeId expectedTy = follow(*expectedType); if (get(expectedTy) || get(expectedTy)) { TypeId ty = arena->addType(BlockedTypeVar{}); TypeId singletonType = arena->addType(SingletonTypeVar(StringSingleton{std::string(string->value.data, string->value.size)})); addConstraint(scope, string->location, PrimitiveTypeConstraint{ty, expectedTy, singletonType, singletonTypes->stringType}); return Inference{ty}; } else if (maybeSingleton(expectedTy)) return Inference{arena->addType(SingletonTypeVar{StringSingleton{std::string{string->value.data, string->value.size}}})}; return Inference{singletonTypes->stringType}; } return Inference{singletonTypes->stringType}; } Inference ConstraintGraphBuilder::check(const ScopePtr& scope, AstExprConstantBool* boolExpr, std::optional expectedType, bool forceSingleton) { const TypeId singletonType = boolExpr->value ? singletonTypes->trueType : singletonTypes->falseType; if (forceSingleton) return Inference{singletonType}; if (expectedType) { const TypeId expectedTy = follow(*expectedType); if (get(expectedTy) || get(expectedTy)) { TypeId ty = arena->addType(BlockedTypeVar{}); addConstraint(scope, boolExpr->location, PrimitiveTypeConstraint{ty, expectedTy, singletonType, singletonTypes->booleanType}); return Inference{ty}; } else if (maybeSingleton(expectedTy)) return Inference{singletonType}; return Inference{singletonTypes->booleanType}; } return Inference{singletonTypes->booleanType}; } Inference ConstraintGraphBuilder::check(const ScopePtr& scope, AstExprLocal* local) { std::optional resultTy; auto def = dfg->getDef(local); if (def) resultTy = scope->lookup(*def); if (!resultTy) { if (auto ty = scope->lookup(local->local)) resultTy = *ty; } if (!resultTy) return Inference{singletonTypes->errorRecoveryType()}; // TODO: replace with ice, locals should never exist before its definition. if (def) return Inference{*resultTy, connectiveArena.proposition(*def, singletonTypes->truthyType)}; else return Inference{*resultTy}; } Inference ConstraintGraphBuilder::check(const ScopePtr& scope, AstExprGlobal* global) { if (std::optional ty = scope->lookup(global->name)) return Inference{*ty}; /* prepopulateGlobalScope() has already added all global functions to the environment by this point, so any * global that is not already in-scope is definitely an unknown symbol. */ reportError(global->location, UnknownSymbol{global->name.value}); return Inference{singletonTypes->errorRecoveryType()}; } Inference ConstraintGraphBuilder::check(const ScopePtr& scope, AstExprIndexName* indexName) { TypeId obj = check(scope, indexName->expr).ty; TypeId result = freshType(scope); TableTypeVar::Props props{{indexName->index.value, Property{result}}}; const std::optional indexer; TableTypeVar ttv{std::move(props), indexer, TypeLevel{}, scope.get(), TableState::Free}; TypeId expectedTableType = arena->addType(std::move(ttv)); addConstraint(scope, indexName->expr->location, SubtypeConstraint{obj, expectedTableType}); return Inference{result}; } Inference ConstraintGraphBuilder::check(const ScopePtr& scope, AstExprIndexExpr* indexExpr) { TypeId obj = check(scope, indexExpr->expr).ty; TypeId indexType = check(scope, indexExpr->index).ty; TypeId result = freshType(scope); TableIndexer indexer{indexType, result}; TypeId tableType = arena->addType(TableTypeVar{TableTypeVar::Props{}, TableIndexer{indexType, result}, TypeLevel{}, scope.get(), TableState::Free}); addConstraint(scope, indexExpr->expr->location, SubtypeConstraint{obj, tableType}); return Inference{result}; } Inference ConstraintGraphBuilder::check(const ScopePtr& scope, AstExprUnary* unary) { auto [operandType, connective] = check(scope, unary->expr); TypeId resultType = arena->addType(BlockedTypeVar{}); addConstraint(scope, unary->location, UnaryConstraint{unary->op, operandType, resultType}); if (unary->op == AstExprUnary::Not) return Inference{resultType, connectiveArena.negation(connective)}; else return Inference{resultType}; } Inference ConstraintGraphBuilder::check(const ScopePtr& scope, AstExprBinary* binary, std::optional expectedType) { auto [leftType, rightType, connective] = checkBinary(scope, binary, expectedType); TypeId resultType = arena->addType(BlockedTypeVar{}); addConstraint(scope, binary->location, BinaryConstraint{binary->op, leftType, rightType, resultType}); return Inference{resultType, std::move(connective)}; } Inference ConstraintGraphBuilder::check(const ScopePtr& scope, AstExprIfElse* ifElse, std::optional expectedType) { check(scope, ifElse->condition); TypeId thenType = check(scope, ifElse->trueExpr, expectedType).ty; TypeId elseType = check(scope, ifElse->falseExpr, expectedType).ty; if (ifElse->hasElse) { TypeId resultType = expectedType ? *expectedType : freshType(scope); addConstraint(scope, ifElse->trueExpr->location, SubtypeConstraint{thenType, resultType}); addConstraint(scope, ifElse->falseExpr->location, SubtypeConstraint{elseType, resultType}); return Inference{resultType}; } return Inference{thenType}; } Inference ConstraintGraphBuilder::check(const ScopePtr& scope, AstExprTypeAssertion* typeAssert) { check(scope, typeAssert->expr, std::nullopt); return Inference{resolveType(scope, typeAssert->annotation)}; } std::tuple ConstraintGraphBuilder::checkBinary( const ScopePtr& scope, AstExprBinary* binary, std::optional expectedType) { if (binary->op == AstExprBinary::And) { auto [leftType, leftConnective] = check(scope, binary->left, expectedType); ScopePtr rightScope = childScope(binary->right, scope); applyRefinements(rightScope, binary->right->location, leftConnective); auto [rightType, rightConnective] = check(rightScope, binary->right, expectedType); return {leftType, rightType, connectiveArena.conjunction(leftConnective, rightConnective)}; } else if (binary->op == AstExprBinary::Or) { auto [leftType, leftConnective] = check(scope, binary->left, expectedType); ScopePtr rightScope = childScope(binary->right, scope); applyRefinements(rightScope, binary->right->location, connectiveArena.negation(leftConnective)); auto [rightType, rightConnective] = check(rightScope, binary->right, expectedType); return {leftType, rightType, connectiveArena.disjunction(leftConnective, rightConnective)}; } else if (auto typeguard = matchTypeGuard(binary)) { TypeId leftType = check(scope, binary->left).ty; TypeId rightType = check(scope, binary->right).ty; std::optional def = dfg->getDef(typeguard->target); if (!def) return {leftType, rightType, nullptr}; TypeId discriminantTy = singletonTypes->neverType; if (typeguard->type == "nil") discriminantTy = singletonTypes->nilType; else if (typeguard->type == "string") discriminantTy = singletonTypes->stringType; else if (typeguard->type == "number") discriminantTy = singletonTypes->numberType; else if (typeguard->type == "boolean") discriminantTy = singletonTypes->threadType; else if (typeguard->type == "table") discriminantTy = singletonTypes->neverType; // TODO: replace with top table type else if (typeguard->type == "function") discriminantTy = singletonTypes->functionType; else if (typeguard->type == "userdata") { // For now, we don't really care about being accurate with userdata if the typeguard was using typeof discriminantTy = singletonTypes->neverType; // TODO: replace with top class type } else if (!typeguard->isTypeof && typeguard->type == "vector") discriminantTy = singletonTypes->neverType; // TODO: figure out a way to deal with this quirky type else if (!typeguard->isTypeof) discriminantTy = singletonTypes->neverType; else if (auto typeFun = globalScope->lookupType(typeguard->type); typeFun && typeFun->typeParams.empty() && typeFun->typePackParams.empty()) { TypeId ty = follow(typeFun->type); // We're only interested in the root class of any classes. if (auto ctv = get(ty); !ctv || !ctv->parent) discriminantTy = ty; } ConnectiveId proposition = connectiveArena.proposition(*def, discriminantTy); if (binary->op == AstExprBinary::CompareEq) return {leftType, rightType, proposition}; else if (binary->op == AstExprBinary::CompareNe) return {leftType, rightType, connectiveArena.negation(proposition)}; else ice->ice("matchTypeGuard should only return a Some under `==` or `~=`!"); } else if (binary->op == AstExprBinary::CompareEq || binary->op == AstExprBinary::CompareNe) { TypeId leftType = check(scope, binary->left, expectedType, true).ty; TypeId rightType = check(scope, binary->right, expectedType, true).ty; ConnectiveId leftConnective = nullptr; if (auto def = dfg->getDef(binary->left)) leftConnective = connectiveArena.proposition(*def, rightType); ConnectiveId rightConnective = nullptr; if (auto def = dfg->getDef(binary->right)) rightConnective = connectiveArena.proposition(*def, leftType); if (binary->op == AstExprBinary::CompareNe) { leftConnective = connectiveArena.negation(leftConnective); rightConnective = connectiveArena.negation(rightConnective); } return {leftType, rightType, connectiveArena.equivalence(leftConnective, rightConnective)}; } else { TypeId leftType = check(scope, binary->left, expectedType).ty; TypeId rightType = check(scope, binary->right, expectedType).ty; return {leftType, rightType, nullptr}; } } TypePackId ConstraintGraphBuilder::checkLValues(const ScopePtr& scope, AstArray exprs) { std::vector types; types.reserve(exprs.size); for (size_t i = 0; i < exprs.size; ++i) { AstExpr* const expr = exprs.data[i]; types.push_back(checkLValue(scope, expr)); } return arena->addTypePack(std::move(types)); } /** * If the expr is a dotted set of names, and if the root symbol refers to an * unsealed table, return that table type, plus the indeces that follow as a * vector. */ static std::optional>> extractDottedName(AstExpr* expr) { std::vector names; while (expr) { if (auto global = expr->as()) { std::reverse(begin(names), end(names)); return std::pair{global->name, std::move(names)}; } else if (auto local = expr->as()) { std::reverse(begin(names), end(names)); return std::pair{local->local, std::move(names)}; } else if (auto indexName = expr->as()) { names.push_back(indexName->index.value); expr = indexName->expr; } else return std::nullopt; } return std::nullopt; } /** * This function is mostly about identifying properties that are being inserted into unsealed tables. * * If expr has the form name.a.b.c */ TypeId ConstraintGraphBuilder::checkLValue(const ScopePtr& scope, AstExpr* expr) { if (auto indexExpr = expr->as()) { if (auto constantString = indexExpr->index->as()) { AstName syntheticIndex{constantString->value.data}; AstExprIndexName synthetic{ indexExpr->location, indexExpr->expr, syntheticIndex, constantString->location, indexExpr->expr->location.end, '.'}; return checkLValue(scope, &synthetic); } } else if (!expr->is()) return check(scope, expr).ty; auto dottedPath = extractDottedName(expr); if (!dottedPath) return check(scope, expr).ty; const auto [sym, segments] = std::move(*dottedPath); LUAU_ASSERT(!segments.empty()); auto lookupResult = scope->lookupEx(sym); if (!lookupResult) return check(scope, expr).ty; const auto [subjectType, symbolScope] = std::move(*lookupResult); TypeId propTy = freshType(scope); std::vector segmentStrings(begin(segments), end(segments)); TypeId updatedType = arena->addType(BlockedTypeVar{}); addConstraint(scope, expr->location, SetPropConstraint{updatedType, subjectType, std::move(segmentStrings), propTy}); std::optional def = dfg->getDef(sym); LUAU_ASSERT(def); symbolScope->bindings[sym].typeId = updatedType; symbolScope->dcrRefinements[*def] = updatedType; astTypes[expr] = propTy; return propTy; } Inference ConstraintGraphBuilder::check(const ScopePtr& scope, AstExprTable* expr, std::optional expectedType) { TypeId ty = arena->addType(TableTypeVar{}); TableTypeVar* ttv = getMutable(ty); LUAU_ASSERT(ttv); ttv->state = TableState::Unsealed; ttv->scope = scope.get(); auto createIndexer = [this, scope, ttv](const Location& location, TypeId currentIndexType, TypeId currentResultType) { if (!ttv->indexer) { TypeId indexType = this->freshType(scope); TypeId resultType = this->freshType(scope); ttv->indexer = TableIndexer{indexType, resultType}; } addConstraint(scope, location, SubtypeConstraint{ttv->indexer->indexType, currentIndexType}); addConstraint(scope, location, SubtypeConstraint{ttv->indexer->indexResultType, currentResultType}); }; for (const AstExprTable::Item& item : expr->items) { std::optional expectedValueType; if (item.key && expectedType) { if (auto stringKey = item.key->as()) { expectedValueType = arena->addType(BlockedTypeVar{}); addConstraint(scope, item.value->location, HasPropConstraint{*expectedValueType, *expectedType, stringKey->value.data}); } } TypeId itemTy = check(scope, item.value, expectedValueType).ty; if (item.key) { // Even though we don't need to use the type of the item's key if // it's a string constant, we still want to check it to populate // astTypes. TypeId keyTy = check(scope, item.key).ty; if (AstExprConstantString* key = item.key->as()) { ttv->props[key->value.begin()] = {itemTy}; } else { createIndexer(item.key->location, keyTy, itemTy); } } else { TypeId numberType = singletonTypes->numberType; // FIXME? The location isn't quite right here. Not sure what is // right. createIndexer(item.value->location, numberType, itemTy); } } return Inference{ty}; } ConstraintGraphBuilder::FunctionSignature ConstraintGraphBuilder::checkFunctionSignature(const ScopePtr& parent, AstExprFunction* fn) { ScopePtr signatureScope = nullptr; ScopePtr bodyScope = nullptr; TypePackId returnType = nullptr; std::vector genericTypes; std::vector genericTypePacks; bool hasGenerics = fn->generics.size > 0 || fn->genericPacks.size > 0; // If we don't have any generics, we can save some memory and compute by not // creating the signatureScope, which is only used to scope the declared // generics properly. if (hasGenerics) { signatureScope = childScope(fn, parent); // We need to assign returnType before creating bodyScope so that the // return type gets propogated to bodyScope. returnType = freshTypePack(signatureScope); signatureScope->returnType = returnType; bodyScope = childScope(fn->body, signatureScope); std::vector> genericDefinitions = createGenerics(signatureScope, fn->generics); std::vector> genericPackDefinitions = createGenericPacks(signatureScope, fn->genericPacks); // We do not support default values on function generics, so we only // care about the types involved. for (const auto& [name, g] : genericDefinitions) { genericTypes.push_back(g.ty); signatureScope->privateTypeBindings[name] = TypeFun{g.ty}; } for (const auto& [name, g] : genericPackDefinitions) { genericTypePacks.push_back(g.tp); signatureScope->privateTypePackBindings[name] = g.tp; } } else { bodyScope = childScope(fn, parent); returnType = freshTypePack(bodyScope); bodyScope->returnType = returnType; // To eliminate the need to branch on hasGenerics below, we say that the // signature scope is the body scope when there is no real signature // scope. signatureScope = bodyScope; } TypePackId varargPack = nullptr; if (fn->vararg) { if (fn->varargAnnotation) { TypePackId annotationType = resolveTypePack(signatureScope, fn->varargAnnotation); varargPack = annotationType; } else { varargPack = arena->freshTypePack(signatureScope.get()); } signatureScope->varargPack = varargPack; } else { varargPack = arena->addTypePack(VariadicTypePack{singletonTypes->anyType, /*hidden*/ true}); // We do not add to signatureScope->varargPack because ... is not valid // in functions without an explicit ellipsis. } LUAU_ASSERT(nullptr != varargPack); if (fn->returnAnnotation) { TypePackId annotatedRetType = resolveTypePack(signatureScope, *fn->returnAnnotation); // We bind the annotated type directly here so that, when we need to // generate constraints for return types, we have a guarantee that we // know the annotated return type already, if one was provided. LUAU_ASSERT(get(returnType)); asMutable(returnType)->ty.emplace(annotatedRetType); } std::vector argTypes; for (AstLocal* local : fn->args) { TypeId t = freshType(signatureScope); argTypes.push_back(t); signatureScope->bindings[local] = Binding{t, local->location}; if (auto def = dfg->getDef(local)) signatureScope->dcrRefinements[*def] = t; if (local->annotation) { TypeId argAnnotation = resolveType(signatureScope, local->annotation, /* topLevel */ true); addConstraint(signatureScope, local->annotation->location, SubtypeConstraint{t, argAnnotation}); } } // TODO: Vararg annotation. // TODO: Preserve argument names in the function's type. FunctionTypeVar actualFunction{TypeLevel{}, parent.get(), arena->addTypePack(argTypes, varargPack), returnType}; actualFunction.hasNoGenerics = !hasGenerics; actualFunction.generics = std::move(genericTypes); actualFunction.genericPacks = std::move(genericTypePacks); TypeId actualFunctionType = arena->addType(std::move(actualFunction)); LUAU_ASSERT(actualFunctionType); astTypes[fn] = actualFunctionType; return { /* signature */ actualFunctionType, // Undo the workaround we made above: if there's no signature scope, // don't report it. /* signatureScope */ hasGenerics ? signatureScope : nullptr, /* bodyScope */ bodyScope, }; } void ConstraintGraphBuilder::checkFunctionBody(const ScopePtr& scope, AstExprFunction* fn) { visitBlockWithoutChildScope(scope, fn->body); // If it is possible for execution to reach the end of the function, the return type must be compatible with () if (nullptr != getFallthrough(fn->body)) { TypePackId empty = arena->addTypePack({}); // TODO we could have CSG retain one of these forever addConstraint(scope, fn->location, PackSubtypeConstraint{scope->returnType, empty}); } } TypeId ConstraintGraphBuilder::resolveType(const ScopePtr& scope, AstType* ty, bool topLevel) { TypeId result = nullptr; if (auto ref = ty->as()) { if (FFlag::DebugLuauMagicTypes) { if (ref->name == "_luau_ice") ice->ice("_luau_ice encountered", ty->location); else if (ref->name == "_luau_print") { if (ref->parameters.size != 1 || !ref->parameters.data[0].type) { reportError(ty->location, GenericError{"_luau_print requires one generic parameter"}); return singletonTypes->errorRecoveryType(); } else return resolveType(scope, ref->parameters.data[0].type, topLevel); } } std::optional alias; if (ref->prefix.has_value()) { alias = scope->lookupImportedType(ref->prefix->value, ref->name.value); } else { alias = scope->lookupType(ref->name.value); } if (alias.has_value()) { // If the alias is not generic, we don't need to set up a blocked // type and an instantiation constraint. if (alias.has_value() && alias->typeParams.empty() && alias->typePackParams.empty()) { result = alias->type; } else { std::vector parameters; std::vector packParameters; for (const AstTypeOrPack& p : ref->parameters) { // We do not enforce the ordering of types vs. type packs here; // that is done in the parser. if (p.type) { parameters.push_back(resolveType(scope, p.type)); } else if (p.typePack) { packParameters.push_back(resolveTypePack(scope, p.typePack)); } else { // This indicates a parser bug: one of these two pointers // should be set. LUAU_ASSERT(false); } } result = arena->addType(PendingExpansionTypeVar{ref->prefix, ref->name, parameters, packParameters}); if (topLevel) { addConstraint(scope, ty->location, TypeAliasExpansionConstraint{/* target */ result}); } } } else { std::string typeName; if (ref->prefix) typeName = std::string(ref->prefix->value) + "."; typeName += ref->name.value; result = singletonTypes->errorRecoveryType(); } } else if (auto tab = ty->as()) { TableTypeVar::Props props; std::optional indexer; for (const AstTableProp& prop : tab->props) { std::string name = prop.name.value; // TODO: Recursion limit. TypeId propTy = resolveType(scope, prop.type); // TODO: Fill in location. props[name] = {propTy}; } if (tab->indexer) { // TODO: Recursion limit. indexer = TableIndexer{ resolveType(scope, tab->indexer->indexType), resolveType(scope, tab->indexer->resultType), }; } result = arena->addType(TableTypeVar{props, indexer, scope->level, scope.get(), TableState::Sealed}); } else if (auto fn = ty->as()) { // TODO: Recursion limit. bool hasGenerics = fn->generics.size > 0 || fn->genericPacks.size > 0; ScopePtr signatureScope = nullptr; std::vector genericTypes; std::vector genericTypePacks; // If we don't have generics, we do not need to generate a child scope // for the generic bindings to live on. if (hasGenerics) { signatureScope = childScope(fn, scope); std::vector> genericDefinitions = createGenerics(signatureScope, fn->generics); std::vector> genericPackDefinitions = createGenericPacks(signatureScope, fn->genericPacks); for (const auto& [name, g] : genericDefinitions) { genericTypes.push_back(g.ty); signatureScope->privateTypeBindings[name] = TypeFun{g.ty}; } for (const auto& [name, g] : genericPackDefinitions) { genericTypePacks.push_back(g.tp); signatureScope->privateTypePackBindings[name] = g.tp; } } else { // To eliminate the need to branch on hasGenerics below, we say that // the signature scope is the parent scope if we don't have // generics. signatureScope = scope; } TypePackId argTypes = resolveTypePack(signatureScope, fn->argTypes); TypePackId returnTypes = resolveTypePack(signatureScope, fn->returnTypes); // TODO: FunctionTypeVar needs a pointer to the scope so that we know // how to quantify/instantiate it. FunctionTypeVar ftv{TypeLevel{}, scope.get(), {}, {}, argTypes, returnTypes}; // This replicates the behavior of the appropriate FunctionTypeVar // constructors. ftv.hasNoGenerics = !hasGenerics; ftv.generics = std::move(genericTypes); ftv.genericPacks = std::move(genericTypePacks); ftv.argNames.reserve(fn->argNames.size); for (const auto& el : fn->argNames) { if (el) { const auto& [name, location] = *el; ftv.argNames.push_back(FunctionArgument{name.value, location}); } else { ftv.argNames.push_back(std::nullopt); } } result = arena->addType(std::move(ftv)); } else if (auto tof = ty->as()) { // TODO: Recursion limit. TypeId exprType = check(scope, tof->expr).ty; result = exprType; } else if (auto unionAnnotation = ty->as()) { std::vector parts; for (AstType* part : unionAnnotation->types) { // TODO: Recursion limit. parts.push_back(resolveType(scope, part, topLevel)); } result = arena->addType(UnionTypeVar{parts}); } else if (auto intersectionAnnotation = ty->as()) { std::vector parts; for (AstType* part : intersectionAnnotation->types) { // TODO: Recursion limit. parts.push_back(resolveType(scope, part, topLevel)); } result = arena->addType(IntersectionTypeVar{parts}); } else if (auto boolAnnotation = ty->as()) { result = arena->addType(SingletonTypeVar(BooleanSingleton{boolAnnotation->value})); } else if (auto stringAnnotation = ty->as()) { result = arena->addType(SingletonTypeVar(StringSingleton{std::string(stringAnnotation->value.data, stringAnnotation->value.size)})); } else if (ty->is()) { result = singletonTypes->errorRecoveryType(); } else { LUAU_ASSERT(0); result = singletonTypes->errorRecoveryType(); } astResolvedTypes[ty] = result; return result; } TypePackId ConstraintGraphBuilder::resolveTypePack(const ScopePtr& scope, AstTypePack* tp) { TypePackId result; if (auto expl = tp->as()) { result = resolveTypePack(scope, expl->typeList); } else if (auto var = tp->as()) { TypeId ty = resolveType(scope, var->variadicType); result = arena->addTypePack(TypePackVar{VariadicTypePack{ty}}); } else if (auto gen = tp->as()) { if (std::optional lookup = scope->lookupPack(gen->genericName.value)) { result = *lookup; } else { reportError(tp->location, UnknownSymbol{gen->genericName.value, UnknownSymbol::Context::Type}); result = singletonTypes->errorRecoveryTypePack(); } } else { LUAU_ASSERT(0); result = singletonTypes->errorRecoveryTypePack(); } astResolvedTypePacks[tp] = result; return result; } TypePackId ConstraintGraphBuilder::resolveTypePack(const ScopePtr& scope, const AstTypeList& list) { std::vector head; for (AstType* headTy : list.types) { head.push_back(resolveType(scope, headTy)); } std::optional tail = std::nullopt; if (list.tailType) { tail = resolveTypePack(scope, list.tailType); } return arena->addTypePack(TypePack{head, tail}); } std::vector> ConstraintGraphBuilder::createGenerics(const ScopePtr& scope, AstArray generics) { std::vector> result; for (const auto& generic : generics) { TypeId genericTy = arena->addType(GenericTypeVar{scope.get(), generic.name.value}); std::optional defaultTy = std::nullopt; if (generic.defaultValue) defaultTy = resolveType(scope, generic.defaultValue); result.push_back({generic.name.value, GenericTypeDefinition{genericTy, defaultTy}}); } return result; } std::vector> ConstraintGraphBuilder::createGenericPacks( const ScopePtr& scope, AstArray generics) { std::vector> result; for (const auto& generic : generics) { TypePackId genericTy = arena->addTypePack(TypePackVar{GenericTypePack{scope.get(), generic.name.value}}); std::optional defaultTy = std::nullopt; if (generic.defaultValue) defaultTy = resolveTypePack(scope, generic.defaultValue); result.push_back({generic.name.value, GenericTypePackDefinition{genericTy, defaultTy}}); } return result; } Inference ConstraintGraphBuilder::flattenPack(const ScopePtr& scope, Location location, InferencePack pack) { const auto& [tp, connectives] = pack; ConnectiveId connective = nullptr; if (!connectives.empty()) connective = connectives[0]; if (auto f = first(tp)) return Inference{*f, connective}; TypeId typeResult = freshType(scope); TypePack onePack{{typeResult}, freshTypePack(scope)}; TypePackId oneTypePack = arena->addTypePack(std::move(onePack)); addConstraint(scope, location, PackSubtypeConstraint{tp, oneTypePack}); return Inference{typeResult, connective}; } void ConstraintGraphBuilder::reportError(Location location, TypeErrorData err) { errors.push_back(TypeError{location, moduleName, std::move(err)}); if (FFlag::DebugLuauLogSolverToJson) logger->captureGenerationError(errors.back()); } void ConstraintGraphBuilder::reportCodeTooComplex(Location location) { errors.push_back(TypeError{location, moduleName, CodeTooComplex{}}); if (FFlag::DebugLuauLogSolverToJson) logger->captureGenerationError(errors.back()); } struct GlobalPrepopulator : AstVisitor { const NotNull globalScope; const NotNull arena; GlobalPrepopulator(NotNull globalScope, NotNull arena) : globalScope(globalScope) , arena(arena) { } bool visit(AstStatFunction* function) override { if (AstExprGlobal* g = function->name->as()) globalScope->bindings[g->name] = Binding{arena->addType(BlockedTypeVar{})}; return true; } }; void ConstraintGraphBuilder::prepopulateGlobalScope(const ScopePtr& globalScope, AstStatBlock* program) { GlobalPrepopulator gp{NotNull{globalScope.get()}, arena}; program->visit(&gp); } std::vector> borrowConstraints(const std::vector& constraints) { std::vector> result; result.reserve(constraints.size()); for (const auto& c : constraints) result.emplace_back(c.get()); return result; } } // namespace Luau