// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details #include "Luau/ConstraintGenerator.h" #include "Luau/Ast.h" #include "Luau/Def.h" #include "Luau/Common.h" #include "Luau/Constraint.h" #include "Luau/ControlFlow.h" #include "Luau/DcrLogger.h" #include "Luau/DenseHash.h" #include "Luau/ModuleResolver.h" #include "Luau/RecursionCounter.h" #include "Luau/Refinement.h" #include "Luau/Scope.h" #include "Luau/TypeUtils.h" #include "Luau/Type.h" #include "Luau/TypeFamily.h" #include "Luau/Simplify.h" #include "Luau/VisitType.h" #include "Luau/InsertionOrderedMap.h" #include LUAU_FASTINT(LuauCheckRecursionLimit); LUAU_FASTFLAG(DebugLuauLogSolverToJson); LUAU_FASTFLAG(DebugLuauMagicTypes); LUAU_FASTFLAG(LuauLoopControlFlowAnalysis); namespace Luau { bool doesCallError(const AstExprCall* call); // TypeInfer.cpp 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), }; } static bool matchAssert(const AstExprCall& call) { if (call.args.size < 1) return false; const AstExprGlobal* funcAsGlobal = call.func->as(); if (!funcAsGlobal || funcAsGlobal->name != "assert") return false; return true; } namespace { struct Checkpoint { size_t offset; }; Checkpoint checkpoint(const ConstraintGenerator* cg) { return Checkpoint{cg->constraints.size()}; } template void forEachConstraint(const Checkpoint& start, const Checkpoint& end, const ConstraintGenerator* cg, F f) { for (size_t i = start.offset; i < end.offset; ++i) f(cg->constraints[i]); } } // namespace ConstraintGenerator::ConstraintGenerator(ModulePtr module, NotNull normalizer, NotNull moduleResolver, NotNull builtinTypes, NotNull ice, const ScopePtr& globalScope, std::function prepareModuleScope, DcrLogger* logger, NotNull dfg, std::vector requireCycles) : module(module) , builtinTypes(builtinTypes) , arena(normalizer->arena) , rootScope(nullptr) , dfg(dfg) , normalizer(normalizer) , moduleResolver(moduleResolver) , ice(ice) , globalScope(globalScope) , prepareModuleScope(std::move(prepareModuleScope)) , requireCycles(std::move(requireCycles)) , logger(logger) { LUAU_ASSERT(module); } void ConstraintGenerator::visitModuleRoot(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); fillInInferredBindings(scope, block); if (logger) logger->captureGenerationModule(module); } TypeId ConstraintGenerator::freshType(const ScopePtr& scope) { return Luau::freshType(arena, builtinTypes, scope.get()); } TypePackId ConstraintGenerator::freshTypePack(const ScopePtr& scope) { FreeTypePack f{scope.get()}; return arena->addTypePack(TypePackVar{std::move(f)}); } ScopePtr ConstraintGenerator::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; } std::optional ConstraintGenerator::lookup(Scope* scope, DefId def, bool prototype) { if (get(def)) return scope->lookup(def); if (auto phi = get(def)) { if (auto found = scope->lookup(def)) return *found; else if (!prototype) return std::nullopt; TypeId res = builtinTypes->neverType; for (DefId operand : phi->operands) { // `scope->lookup(operand)` may return nothing because we only bind a type to that operand // once we've seen that particular `DefId`. In this case, we need to prototype those types // and use those at a later time. std::optional ty = lookup(scope, operand, /*prototype*/false); if (!ty) { ty = arena->addType(BlockedType{}); rootScope->lvalueTypes[operand] = *ty; } res = simplifyUnion(builtinTypes, arena, res, *ty).result; } scope->lvalueTypes[def] = res; return res; } else ice->ice("ConstraintGenerator::lookup is inexhaustive?"); } NotNull ConstraintGenerator::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 ConstraintGenerator::addConstraint(const ScopePtr& scope, std::unique_ptr c) { return NotNull{constraints.emplace_back(std::move(c)).get()}; } void ConstraintGenerator::unionRefinements(const RefinementContext& lhs, const RefinementContext& rhs, RefinementContext& dest, std::vector* constraints) { const auto intersect = [&](const std::vector& types) { if (1 == types.size()) return types[0]; else if (2 == types.size()) { // TODO: It may be advantageous to introduce a refine type family here when there are blockedTypes. SimplifyResult sr = simplifyIntersection(builtinTypes, arena, types[0], types[1]); if (sr.blockedTypes.empty()) return sr.result; } return arena->addType(IntersectionType{types}); }; for (auto& [def, partition] : lhs) { auto rhsIt = rhs.find(def); if (rhsIt == rhs.end()) continue; LUAU_ASSERT(!partition.discriminantTypes.empty()); LUAU_ASSERT(!rhsIt->second.discriminantTypes.empty()); TypeId leftDiscriminantTy = partition.discriminantTypes.size() == 1 ? partition.discriminantTypes[0] : intersect(partition.discriminantTypes); TypeId rightDiscriminantTy = rhsIt->second.discriminantTypes.size() == 1 ? rhsIt->second.discriminantTypes[0] : intersect(rhsIt->second.discriminantTypes); dest.insert(def, {}); dest.get(def)->discriminantTypes.push_back(simplifyUnion(builtinTypes, arena, leftDiscriminantTy, rightDiscriminantTy).result); dest.get(def)->shouldAppendNilType |= partition.shouldAppendNilType || rhsIt->second.shouldAppendNilType; } } void ConstraintGenerator::computeRefinement(const ScopePtr& scope, RefinementId refinement, RefinementContext* refis, bool sense, bool eq, std::vector* constraints) { if (!refinement) return; else if (auto variadic = get(refinement)) { for (RefinementId refi : variadic->refinements) computeRefinement(scope, refi, refis, sense, eq, constraints); } else if (auto negation = get(refinement)) return computeRefinement(scope, negation->refinement, refis, !sense, eq, constraints); else if (auto conjunction = get(refinement)) { RefinementContext lhsRefis; RefinementContext rhsRefis; computeRefinement(scope, conjunction->lhs, sense ? refis : &lhsRefis, sense, eq, constraints); computeRefinement(scope, conjunction->rhs, sense ? refis : &rhsRefis, sense, eq, constraints); if (!sense) unionRefinements(lhsRefis, rhsRefis, *refis, constraints); } else if (auto disjunction = get(refinement)) { RefinementContext lhsRefis; RefinementContext rhsRefis; computeRefinement(scope, disjunction->lhs, sense ? &lhsRefis : refis, sense, eq, constraints); computeRefinement(scope, disjunction->rhs, sense ? &rhsRefis : refis, sense, eq, constraints); if (sense) unionRefinements(lhsRefis, rhsRefis, *refis, constraints); } else if (auto equivalence = get(refinement)) { computeRefinement(scope, equivalence->lhs, refis, sense, true, constraints); computeRefinement(scope, equivalence->rhs, refis, sense, true, constraints); } else if (auto proposition = get(refinement)) { TypeId discriminantTy = proposition->discriminantTy; if (!sense && !eq) discriminantTy = arena->addType(NegationType{proposition->discriminantTy}); else if (eq) { discriminantTy = arena->addType(BlockedType{}); constraints->push_back(SingletonOrTopTypeConstraint{discriminantTy, proposition->discriminantTy, !sense}); } for (const RefinementKey* key = proposition->key; key; key = key->parent) { refis->insert(key->def, {}); refis->get(key->def)->discriminantTypes.push_back(discriminantTy); // Reached leaf node if (!key->propName) break; TypeId nextDiscriminantTy = arena->addType(TableType{}); NotNull table{getMutable(nextDiscriminantTy)}; table->props[*key->propName] = {discriminantTy}; table->scope = scope.get(); table->state = TableState::Sealed; discriminantTy = nextDiscriminantTy; } // When the top-level expression is `t[x]`, we want to refine it into `nil`, not `never`. LUAU_ASSERT(refis->get(proposition->key->def)); refis->get(proposition->key->def)->shouldAppendNilType = (sense || !eq) && containsSubscriptedDefinition(proposition->key->def); } } namespace { /* * Constraint generation may be called upon to simplify an intersection or union * of types that are not sufficiently solved yet. We use * FindSimplificationBlockers to recognize these types and defer the * simplification until constraint solution. */ struct FindSimplificationBlockers : TypeOnceVisitor { bool found = false; bool visit(TypeId) override { return !found; } bool visit(TypeId, const BlockedType&) override { found = true; return false; } bool visit(TypeId, const FreeType&) override { found = true; return false; } bool visit(TypeId, const PendingExpansionType&) override { found = true; return false; } // We do not need to know anything at all about a function's argument or // return types in order to simplify it in an intersection or union. bool visit(TypeId, const FunctionType&) override { return false; } bool visit(TypeId, const ClassType&) override { return false; } }; bool mustDeferIntersection(TypeId ty) { FindSimplificationBlockers bts; bts.traverse(ty); return bts.found; } } // namespace void ConstraintGenerator::applyRefinements(const ScopePtr& scope, Location location, RefinementId refinement) { if (!refinement) return; RefinementContext refinements; std::vector constraints; computeRefinement(scope, refinement, &refinements, /*sense*/ true, /*eq*/ false, &constraints); for (auto& [def, partition] : refinements) { if (std::optional defTy = lookup(scope.get(), def)) { TypeId ty = *defTy; if (partition.shouldAppendNilType) ty = arena->addType(UnionType{{ty, builtinTypes->nilType}}); // Intersect ty with every discriminant type. If either type is not // sufficiently solved, we queue the intersection up via an // IntersectConstraint. for (TypeId dt : partition.discriminantTypes) { if (mustDeferIntersection(ty) || mustDeferIntersection(dt)) { TypeId resultType = arena->addType(TypeFamilyInstanceType{ NotNull{&kBuiltinTypeFamilies.refineFamily}, {ty, dt}, {}, }); addConstraint(scope, location, ReduceConstraint{resultType}); ty = resultType; } else { switch (shouldSuppressErrors(normalizer, ty)) { case ErrorSuppression::DoNotSuppress: ty = simplifyIntersection(builtinTypes, arena, ty, dt).result; break; case ErrorSuppression::Suppress: ty = simplifyIntersection(builtinTypes, arena, ty, dt).result; ty = simplifyUnion(builtinTypes, arena, ty, builtinTypes->errorType).result; break; case ErrorSuppression::NormalizationFailed: reportError(location, NormalizationTooComplex{}); ty = simplifyIntersection(builtinTypes, arena, ty, dt).result; break; } } } scope->rvalueRefinements[def] = ty; } } for (auto& c : constraints) addConstraint(scope, location, c); } ControlFlow ConstraintGenerator::visitBlockWithoutChildScope(const ScopePtr& scope, AstStatBlock* block) { RecursionCounter counter{&recursionCount}; if (recursionCount >= FInt::LuauCheckRecursionLimit) { reportCodeTooComplex(block->location); return ControlFlow::None; } 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. for (AstStat* stat : block->body) { if (auto alias = stat->as()) { if (scope->exportedTypeBindings.count(alias->name.value) || scope->privateTypeBindings.count(alias->name.value)) { auto it = aliasDefinitionLocations.find(alias->name.value); LUAU_ASSERT(it != aliasDefinitionLocations.end()); reportError(alias->location, DuplicateTypeDefinition{alias->name.value, it->second}); continue; } // A type alias might have no name if the code is syntactically // illegal. We mustn't prepopulate anything in this case. if (alias->name == kParseNameError || alias->name == "typeof") continue; ScopePtr defnScope = childScope(alias, scope); TypeId initialType = arena->addType(BlockedType{}); TypeFun initialFun{initialType}; for (const auto& [name, gen] : createGenerics(defnScope, alias->generics, /* useCache */ true)) { initialFun.typeParams.push_back(gen); } for (const auto& [name, genPack] : createGenericPacks(defnScope, alias->genericPacks, /* useCache */ true)) { initialFun.typePackParams.push_back(genPack); } if (alias->exported) scope->exportedTypeBindings[alias->name.value] = std::move(initialFun); else scope->privateTypeBindings[alias->name.value] = std::move(initialFun); astTypeAliasDefiningScopes[alias] = defnScope; aliasDefinitionLocations[alias->name.value] = alias->location; } } std::optional firstControlFlow; for (AstStat* stat : block->body) { ControlFlow cf = visit(scope, stat); if (cf != ControlFlow::None && !firstControlFlow) firstControlFlow = cf; } return firstControlFlow.value_or(ControlFlow::None); } ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStat* stat) { RecursionLimiter limiter{&recursionCount, FInt::LuauCheckRecursionLimit}; if (auto s = stat->as()) return visit(scope, s); else if (auto i = stat->as()) return visit(scope, i); else if (auto s = stat->as()) return visit(scope, s); else if (auto s = stat->as()) return visit(scope, s); else if (stat->is()) return FFlag::LuauLoopControlFlowAnalysis ? ControlFlow::Breaks : ControlFlow::None; else if (stat->is()) return FFlag::LuauLoopControlFlowAnalysis ? ControlFlow::Continues : ControlFlow::None; else if (auto r = stat->as()) return visit(scope, r); else if (auto e = stat->as()) { checkPack(scope, e->expr); if (auto call = e->expr->as(); call && doesCallError(call)) return ControlFlow::Throws; return ControlFlow::None; } else if (auto s = stat->as()) return visit(scope, s); else if (auto s = stat->as()) return visit(scope, s); else if (auto s = stat->as()) return visit(scope, s); else if (auto a = stat->as()) return visit(scope, a); else if (auto a = stat->as()) return visit(scope, a); else if (auto f = stat->as()) return visit(scope, f); else if (auto f = stat->as()) return visit(scope, f); else if (auto a = stat->as()) return visit(scope, a); else if (auto s = stat->as()) return visit(scope, s); else if (auto s = stat->as()) return visit(scope, s); else if (auto s = stat->as()) return visit(scope, s); else if (auto s = stat->as()) return visit(scope, s); else { LUAU_ASSERT(0 && "Internal error: Unknown AstStat type"); return ControlFlow::None; } } ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatLocal* statLocal) { std::vector annotatedTypes; annotatedTypes.reserve(statLocal->vars.size); bool hasAnnotation = false; std::vector> expectedTypes; expectedTypes.reserve(statLocal->vars.size); std::vector assignees; assignees.reserve(statLocal->vars.size); // Used to name the first value type, even if it's not placed in varTypes, // for the purpose of synthetic name attribution. std::optional firstValueType; for (AstLocal* local : statLocal->vars) { const Location location = local->location; TypeId assignee = arena->addType(LocalType{builtinTypes->neverType, /* blockCount */ 1, local->name.value}); assignees.push_back(assignee); if (!firstValueType) firstValueType = assignee; if (local->annotation) { hasAnnotation = true; TypeId annotationTy = resolveType(scope, local->annotation, /* inTypeArguments */ false); annotatedTypes.push_back(annotationTy); expectedTypes.push_back(annotationTy); scope->bindings[local] = Binding{annotationTy, location}; } else { // annotatedTypes must contain one type per local. If a particular // local has no annotation at, assume the most conservative thing. annotatedTypes.push_back(builtinTypes->unknownType); expectedTypes.push_back(std::nullopt); scope->bindings[local] = Binding{builtinTypes->unknownType, location}; inferredBindings[local] = {scope.get(), location, {assignee}}; } DefId def = dfg->getDef(local); scope->lvalueTypes[def] = assignee; } TypePackId resultPack = checkPack(scope, statLocal->values, expectedTypes).tp; addConstraint(scope, statLocal->location, UnpackConstraint{arena->addTypePack(std::move(assignees)), resultPack, /*resultIsLValue*/ true}); // Types must flow between whatever annotations were provided and the rhs expression. if (hasAnnotation) addConstraint(scope, statLocal->location, PackSubtypeConstraint{resultPack, arena->addTypePack(std::move(annotatedTypes))}); if (statLocal->vars.size == 1 && statLocal->values.size == 1 && firstValueType && scope.get() == rootScope) { AstLocal* var = statLocal->vars.data[0]; AstExpr* value = statLocal->values.data[0]; if (value->is()) addConstraint(scope, value->location, NameConstraint{*firstValueType, var->name.value, /*synthetic*/ true}); else if (const AstExprCall* call = value->as()) { if (const AstExprGlobal* global = call->func->as(); global && global->name == "setmetatable") { addConstraint(scope, value->location, NameConstraint{*firstValueType, var->name.value, /*synthetic*/ true}); } } } if (statLocal->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 < statLocal->values.size && i < statLocal->vars.size; ++i) { const AstExprCall* call = statLocal->values.data[i]->as(); if (!call) continue; auto maybeRequire = matchRequire(*call); if (!maybeRequire) continue; AstExpr* require = *maybeRequire; auto moduleInfo = moduleResolver->resolveModuleInfo(module->name, *require); if (!moduleInfo) continue; ModulePtr module = moduleResolver->getModule(moduleInfo->name); if (!module) continue; const Name name{statLocal->vars.data[i]->name.value}; scope->importedTypeBindings[name] = module->exportedTypeBindings; scope->importedModules[name] = moduleInfo->name; // Imported types of requires that transitively refer to current module have to be replaced with 'any' for (const auto& [location, path] : requireCycles) { if (path.empty() || path.front() != moduleInfo->name) continue; for (auto& [name, tf] : scope->importedTypeBindings[name]) tf = TypeFun{{}, {}, builtinTypes->anyType}; } } } return ControlFlow::None; } ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatFor* for_) { TypeId annotationTy = builtinTypes->numberType; if (for_->var->annotation) annotationTy = resolveType(scope, for_->var->annotation, /* inTypeArguments */ false); auto inferNumber = [&](AstExpr* expr) { if (!expr) return; TypeId t = check(scope, expr).ty; addConstraint(scope, expr->location, SubtypeConstraint{t, builtinTypes->numberType}); }; inferNumber(for_->from); inferNumber(for_->to); inferNumber(for_->step); ScopePtr forScope = childScope(for_, scope); forScope->bindings[for_->var] = Binding{annotationTy, for_->var->location}; DefId def = dfg->getDef(for_->var); forScope->lvalueTypes[def] = annotationTy; forScope->rvalueRefinements[def] = annotationTy; visit(forScope, for_->body); return ControlFlow::None; } ControlFlow ConstraintGenerator::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 assignee = arena->addType(BlockedType{}); variableTypes.push_back(assignee); if (var->annotation) { TypeId annotationTy = resolveType(loopScope, var->annotation, /*inTypeArguments*/ false); loopScope->bindings[var] = Binding{annotationTy, var->location}; addConstraint(scope, var->location, SubtypeConstraint{assignee, annotationTy}); } else loopScope->bindings[var] = Binding{assignee, var->location}; DefId def = dfg->getDef(var); loopScope->lvalueTypes[def] = assignee; } TypePackId variablePack = arena->addTypePack(std::move(variableTypes)); addConstraint( loopScope, getLocation(forIn->values), IterableConstraint{iterator, variablePack, forIn->values.data[0], &module->astForInNextTypes}); visit(loopScope, forIn->body); return ControlFlow::None; } ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatWhile* while_) { RefinementId refinement = check(scope, while_->condition).refinement; ScopePtr whileScope = childScope(while_, scope); applyRefinements(whileScope, while_->condition->location, refinement); visit(whileScope, while_->body); return ControlFlow::None; } ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatRepeat* repeat) { ScopePtr repeatScope = childScope(repeat, scope); visitBlockWithoutChildScope(repeatScope, repeat->body); check(repeatScope, repeat->condition); return ControlFlow::None; } ControlFlow ConstraintGenerator::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(BlockedType{}); scope->bindings[function->name] = Binding{functionType, function->name->location}; FunctionSignature sig = checkFunctionSignature(scope, function->func, /* expectedType */ std::nullopt, function->name->location); sig.bodyScope->bindings[function->name] = Binding{sig.signature, function->func->location}; DefId def = dfg->getDef(function->name); scope->lvalueTypes[def] = functionType; scope->rvalueRefinements[def] = functionType; sig.bodyScope->lvalueTypes[def] = sig.signature; sig.bodyScope->rvalueRefinements[def] = sig.signature; Checkpoint start = checkpoint(this); checkFunctionBody(sig.bodyScope, function->func); Checkpoint 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}); Constraint* previous = nullptr; forEachConstraint(start, end, this, [&c, &previous](const ConstraintPtr& constraint) { c->dependencies.push_back(NotNull{constraint.get()}); if (auto psc = get(*constraint); psc && psc->returns) { if (previous) constraint->dependencies.push_back(NotNull{previous}); previous = constraint.get(); } }); addConstraint(scope, std::move(c)); module->astTypes[function->func] = functionType; return ControlFlow::None; } ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatFunction* function) { // Name could be AstStatLocal, AstStatGlobal, AstStatIndexName. // With or without self TypeId generalizedType = arena->addType(BlockedType{}); Checkpoint start = checkpoint(this); FunctionSignature sig = checkFunctionSignature(scope, function->func, /* expectedType */ std::nullopt, function->name->location); DenseHashSet excludeList{nullptr}; DefId def = dfg->getDef(function->name); std::optional existingFunctionTy = lookup(scope.get(), def); if (AstExprLocal* localName = function->name->as()) { if (existingFunctionTy) { addConstraint(scope, function->name->location, SubtypeConstraint{generalizedType, *existingFunctionTy}); Symbol sym{localName->local}; scope->bindings[sym].typeId = generalizedType; } else scope->bindings[localName->local] = Binding{generalizedType, localName->location}; sig.bodyScope->bindings[localName->local] = Binding{sig.signature, localName->location}; sig.bodyScope->lvalueTypes[def] = sig.signature; sig.bodyScope->rvalueRefinements[def] = sig.signature; } else if (AstExprGlobal* globalName = function->name->as()) { if (!existingFunctionTy) ice->ice("prepopulateGlobalScope did not populate a global name", globalName->location); generalizedType = *existingFunctionTy; sig.bodyScope->bindings[globalName->name] = Binding{sig.signature, globalName->location}; sig.bodyScope->lvalueTypes[def] = sig.signature; sig.bodyScope->rvalueRefinements[def] = sig.signature; } else if (AstExprIndexName* indexName = function->name->as()) { Checkpoint check1 = checkpoint(this); std::optional lvalueType = checkLValue(scope, indexName, generalizedType); LUAU_ASSERT(lvalueType); Checkpoint check2 = checkpoint(this); forEachConstraint(check1, check2, this, [&excludeList](const ConstraintPtr& c) { excludeList.insert(c.get()); }); // TODO figure out how to populate the location field of the table Property. if (lvalueType && *lvalueType != generalizedType) { addConstraint(scope, indexName->location, SubtypeConstraint{*lvalueType, generalizedType}); } } else if (AstExprError* err = function->name->as()) { generalizedType = builtinTypes->errorRecoveryType(); } if (generalizedType == nullptr) ice->ice("generalizedType == nullptr", function->location); scope->rvalueRefinements[def] = generalizedType; checkFunctionBody(sig.bodyScope, function->func); Checkpoint 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{generalizedType, sig.signature}); Constraint* previous = nullptr; forEachConstraint(start, end, this, [&c, &excludeList, &previous](const ConstraintPtr& constraint) { if (!excludeList.contains(constraint.get())) c->dependencies.push_back(NotNull{constraint.get()}); if (auto psc = get(*constraint); psc && psc->returns) { if (previous) constraint->dependencies.push_back(NotNull{previous}); previous = constraint.get(); } }); addConstraint(scope, std::move(c)); return ControlFlow::None; } ControlFlow ConstraintGenerator::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, /*returns*/ true}); return ControlFlow::Returns; } ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatBlock* block) { ScopePtr innerScope = childScope(block, scope); ControlFlow flow = visitBlockWithoutChildScope(innerScope, block); // An AstStatBlock has linear control flow, i.e. one entry and one exit, so we can inherit // all the changes to the environment occurred by the statements in that block. scope->inheritRefinements(innerScope); scope->inheritAssignments(innerScope); return flow; } // TODO Clip? static void bindFreeType(TypeId a, TypeId b) { FreeType* af = getMutable(a); FreeType* bf = getMutable(b); LUAU_ASSERT(af || bf); if (!bf) asMutable(a)->ty.emplace(b); else if (!af) asMutable(b)->ty.emplace(a); else if (subsumes(bf->scope, af->scope)) asMutable(a)->ty.emplace(b); else if (subsumes(af->scope, bf->scope)) asMutable(b)->ty.emplace(a); } ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatAssign* assign) { std::vector assignees; assignees.reserve(assign->vars.size); for (AstExpr* lvalue : assign->vars) { TypeId assignee = arena->addType(BlockedType{}); checkLValue(scope, lvalue, assignee); assignees.push_back(assignee); } TypePackId resultPack = checkPack(scope, assign->values).tp; addConstraint(scope, assign->location, UnpackConstraint{arena->addTypePack(std::move(assignees)), resultPack, /*resultIsLValue*/ true}); return ControlFlow::None; } ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatCompoundAssign* assign) { AstExprBinary binop = AstExprBinary{assign->location, assign->op, assign->var, assign->value}; TypeId resultTy = check(scope, &binop).ty; checkLValue(scope, assign->var, resultTy); DefId def = dfg->getDef(assign->var); scope->lvalueTypes[def] = resultTy; return ControlFlow::None; } ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatIf* ifStatement) { RefinementId refinement = check(scope, ifStatement->condition, std::nullopt).refinement; ScopePtr thenScope = childScope(ifStatement->thenbody, scope); applyRefinements(thenScope, ifStatement->condition->location, refinement); ScopePtr elseScope = childScope(ifStatement->elsebody ? ifStatement->elsebody : ifStatement, scope); applyRefinements(elseScope, ifStatement->elseLocation.value_or(ifStatement->condition->location), refinementArena.negation(refinement)); ControlFlow thencf = visit(thenScope, ifStatement->thenbody); ControlFlow elsecf = ControlFlow::None; if (ifStatement->elsebody) elsecf = visit(elseScope, ifStatement->elsebody); if (thencf != ControlFlow::None && elsecf == ControlFlow::None) scope->inheritRefinements(elseScope); else if (thencf == ControlFlow::None && elsecf != ControlFlow::None) scope->inheritRefinements(thenScope); if (thencf == ControlFlow::None) scope->inheritAssignments(thenScope); if (elsecf == ControlFlow::None) scope->inheritAssignments(elseScope); if (FFlag::LuauLoopControlFlowAnalysis && thencf == elsecf) return thencf; else if (matches(thencf, ControlFlow::Returns | ControlFlow::Throws) && matches(elsecf, ControlFlow::Returns | ControlFlow::Throws)) return ControlFlow::Returns; else return ControlFlow::None; } static bool occursCheck(TypeId needle, TypeId haystack) { LUAU_ASSERT(get(needle)); haystack = follow(haystack); auto checkHaystack = [needle](TypeId haystack) { return occursCheck(needle, haystack); }; if (needle == haystack) return true; else if (auto ut = get(haystack)) return std::any_of(begin(ut), end(ut), checkHaystack); else if (auto it = get(haystack)) return std::any_of(begin(it), end(it), checkHaystack); return false; } ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatTypeAlias* alias) { if (alias->name == kParseNameError) return ControlFlow::None; if (alias->name == "typeof") { reportError(alias->location, GenericError{"Type aliases cannot be named typeof"}); return ControlFlow::None; } ScopePtr* defnScope = astTypeAliasDefiningScopes.find(alias); std::unordered_map* typeBindings; if (alias->exported) typeBindings = &scope->exportedTypeBindings; else typeBindings = &scope->privateTypeBindings; // These will be undefined if the alias was a duplicate definition, in which // case we just skip over it. auto bindingIt = typeBindings->find(alias->name.value); if (bindingIt == typeBindings->end() || defnScope == nullptr) return ControlFlow::None; TypeId ty = resolveType(*defnScope, alias->type, /* inTypeArguments */ false); TypeId aliasTy = bindingIt->second.type; LUAU_ASSERT(get(aliasTy)); if (occursCheck(aliasTy, ty)) { asMutable(aliasTy)->ty.emplace(builtinTypes->anyType); reportError(alias->nameLocation, OccursCheckFailed{}); } else asMutable(aliasTy)->ty.emplace(ty); std::vector typeParams; for (auto tyParam : createGenerics(*defnScope, alias->generics, /* useCache */ true, /* addTypes */ false)) typeParams.push_back(tyParam.second.ty); std::vector typePackParams; for (auto tpParam : createGenericPacks(*defnScope, alias->genericPacks, /* useCache */ true, /* addTypes */ false)) typePackParams.push_back(tpParam.second.tp); addConstraint(scope, alias->type->location, NameConstraint{ ty, alias->name.value, /*synthetic=*/false, std::move(typeParams), std::move(typePackParams), }); return ControlFlow::None; } ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatDeclareGlobal* global) { LUAU_ASSERT(global->type); TypeId globalTy = resolveType(scope, global->type, /* inTypeArguments */ false); Name globalName(global->name.value); module->declaredGlobals[globalName] = globalTy; rootScope->bindings[global->name] = Binding{globalTy, global->location}; DefId def = dfg->getDef(global); rootScope->lvalueTypes[def] = globalTy; rootScope->rvalueRefinements[def] = globalTy; return ControlFlow::None; } 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" || name == "__idiv"; } ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatDeclareClass* declaredClass) { std::optional superTy = std::make_optional(builtinTypes->classType); 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 ControlFlow::None; } // 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 ControlFlow::None; } } Name className(declaredClass->name.value); TypeId classTy = arena->addType(ClassType(className, {}, superTy, std::nullopt, {}, {}, module->name)); ClassType* ctv = getMutable(classTy); TypeId metaTy = arena->addType(TableType{TableState::Sealed, scope->level, scope.get()}); TableType* metatable = getMutable(metaTy); ctv->metatable = metaTy; scope->exportedTypeBindings[className] = TypeFun{{}, classTy}; if (declaredClass->indexer) { RecursionCounter counter{&recursionCount}; if (recursionCount >= FInt::LuauCheckRecursionLimit) { reportCodeTooComplex(declaredClass->indexer->location); } else { ctv->indexer = TableIndexer{ resolveType(scope, declaredClass->indexer->indexType, /* inTypeArguments */ false), resolveType(scope, declaredClass->indexer->resultType, /* inTypeArguments */ false), }; } } for (const AstDeclaredClassProp& prop : declaredClass->props) { Name propName(prop.name.value); TypeId propTy = resolveType(scope, prop.ty, /* inTypeArguments */ false); 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 (FunctionType* 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 IntersectionType* itv = get(currentTy)) { std::vector options = itv->parts; options.push_back(propTy); TypeId newItv = arena->addType(IntersectionType{std::move(options)}); if (assignToMetatable) metatable->props[propName] = {newItv}; else ctv->props[propName] = {newItv}; } else if (get(currentTy)) { TypeId intersection = arena->addType(IntersectionType{{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())}); } } } return ControlFlow::None; } ControlFlow ConstraintGenerator::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); } std::vector genericTps; genericTps.reserve(genericPacks.size()); for (auto& [name, generic] : genericPacks) { genericTps.push_back(generic.tp); } ScopePtr funScope = scope; if (!generics.empty() || !genericPacks.empty()) funScope = childScope(global, scope); TypePackId paramPack = resolveTypePack(funScope, global->params, /* inTypeArguments */ false); TypePackId retPack = resolveTypePack(funScope, global->retTypes, /* inTypeArguments */ false); TypeId fnType = arena->addType(FunctionType{TypeLevel{}, funScope.get(), std::move(genericTys), std::move(genericTps), paramPack, retPack}); FunctionType* ftv = getMutable(fnType); ftv->isCheckedFunction = global->checkedFunction; 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}; DefId def = dfg->getDef(global); rootScope->lvalueTypes[def] = fnType; rootScope->rvalueRefinements[def] = fnType; return ControlFlow::None; } ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatError* error) { for (AstStat* stat : error->statements) visit(scope, stat); for (AstExpr* expr : error->expressions) check(scope, expr); return ControlFlow::None; } InferencePack ConstraintGenerator::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, expectedType).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 ConstraintGenerator::checkPack( const ScopePtr& scope, AstExpr* expr, const std::vector>& expectedTypes, bool generalize) { RecursionCounter counter{&recursionCount}; if (recursionCount >= FInt::LuauCheckRecursionLimit) { reportCodeTooComplex(expr->location); return InferencePack{builtinTypes->errorRecoveryTypePack()}; } InferencePack result; if (AstExprCall* call = expr->as()) result = checkPack(scope, call); else if (AstExprVarargs* varargs = expr->as()) { if (scope->varargPack) result = InferencePack{*scope->varargPack}; else result = InferencePack{builtinTypes->errorRecoveryTypePack()}; } else { std::optional expectedType; if (!expectedTypes.empty()) expectedType = expectedTypes[0]; TypeId t = check(scope, expr, expectedType, /*forceSingletons*/ false, generalize).ty; result = InferencePack{arena->addTypePack({t})}; } LUAU_ASSERT(result.tp); module->astTypePacks[expr] = result.tp; return result; } InferencePack ConstraintGenerator::checkPack(const ScopePtr& scope, AstExprCall* call) { std::vector exprArgs; std::vector returnRefinements; std::vector> discriminantTypes; if (call->self) { AstExprIndexName* indexExpr = call->func->as(); if (!indexExpr) ice->ice("method call expression has no 'self'"); exprArgs.push_back(indexExpr->expr); if (auto key = dfg->getRefinementKey(indexExpr->expr)) { TypeId discriminantTy = arena->addType(BlockedType{}); returnRefinements.push_back(refinementArena.proposition(key, discriminantTy)); discriminantTypes.push_back(discriminantTy); } else discriminantTypes.push_back(std::nullopt); } for (AstExpr* arg : call->args) { exprArgs.push_back(arg); if (auto key = dfg->getRefinementKey(arg)) { TypeId discriminantTy = arena->addType(BlockedType{}); returnRefinements.push_back(refinementArena.proposition(key, discriminantTy)); discriminantTypes.push_back(discriminantTy); } else discriminantTypes.push_back(std::nullopt); } TypeId fnType = check(scope, call->func).ty; std::vector> expectedTypesForCall = getExpectedCallTypesForFunctionOverloads(fnType); module->astOriginalCallTypes[call->func] = fnType; module->astOriginalCallTypes[call] = fnType; Checkpoint argBeginCheckpoint = checkpoint(this); std::vector args; std::optional argTail; std::vector argumentRefinements; for (size_t i = 0; i < exprArgs.size(); ++i) { AstExpr* arg = exprArgs[i]; if (i == 0 && call->self) { // The self type has already been computed as a side effect of // computing fnType. If computing that did not cause us to exceed a // recursion limit, we can fetch it from astTypes rather than // recomputing it. TypeId* selfTy = module->astTypes.find(exprArgs[0]); if (selfTy) args.push_back(*selfTy); else args.push_back(freshType(scope)); } else if (i < exprArgs.size() - 1 || !(arg->is() || arg->is())) { auto [ty, refinement] = check(scope, arg, /*expectedType*/ std::nullopt, /*forceSingleton*/ false, /*generalize*/ false); args.push_back(ty); argumentRefinements.push_back(refinement); } else { auto [tp, refis] = checkPack(scope, arg, {}); argTail = tp; argumentRefinements.insert(argumentRefinements.end(), refis.begin(), refis.end()); } } Checkpoint argEndCheckpoint = checkpoint(this); if (matchSetmetatable(*call)) { TypePack argTailPack; if (argTail && args.size() < 2) argTailPack = extendTypePack(*arena, builtinTypes, *argTail, 2 - args.size()); TypeId target = nullptr; TypeId mt = nullptr; if (args.size() + argTailPack.head.size() == 2) { target = args.size() > 0 ? args[0] : argTailPack.head[0]; mt = args.size() > 1 ? args[1] : argTailPack.head[args.size() == 0 ? 1 : 0]; } else { std::vector unpackedTypes; if (args.size() > 0) target = args[0]; else { target = arena->addType(BlockedType{}); unpackedTypes.emplace_back(target); } mt = arena->addType(BlockedType{}); unpackedTypes.emplace_back(mt); TypePackId mtPack = arena->addTypePack(std::move(unpackedTypes)); addConstraint(scope, call->location, UnpackConstraint{mtPack, *argTail}); } LUAU_ASSERT(target); LUAU_ASSERT(mt); target = follow(target); AstExpr* targetExpr = call->args.data[0]; TypeId resultTy = nullptr; if (isTableUnion(target)) { const UnionType* targetUnion = get(target); std::vector newParts; for (TypeId ty : targetUnion) newParts.push_back(arena->addType(MetatableType{ty, mt})); resultTy = arena->addType(UnionType{std::move(newParts)}); } else resultTy = arena->addType(MetatableType{target, mt}); if (AstExprLocal* targetLocal = targetExpr->as()) { scope->bindings[targetLocal->local].typeId = resultTy; DefId def = dfg->getDef(targetLocal); scope->lvalueTypes[def] = resultTy; // TODO: typestates: track this as an assignment scope->rvalueRefinements[def] = resultTy; // TODO: typestates: track this as an assignment recordInferredBinding(targetLocal->local, resultTy); } return InferencePack{arena->addTypePack({resultTy}), {refinementArena.variadic(returnRefinements)}}; } else { if (matchAssert(*call) && !argumentRefinements.empty()) applyRefinements(scope, call->args.data[0]->location, argumentRefinements[0]); // TODO: How do expectedTypes play into this? Do they? TypePackId rets = arena->addTypePack(BlockedTypePack{}); TypePackId argPack = arena->addTypePack(TypePack{args, argTail}); FunctionType ftv(TypeLevel{}, scope.get(), argPack, rets, std::nullopt, call->self); NotNull fcc = addConstraint(scope, call->func->location, FunctionCallConstraint{ fnType, argPack, rets, call, std::move(discriminantTypes), &module->astOverloadResolvedTypes, }); // 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(argBeginCheckpoint, argEndCheckpoint, this, [fcc](const ConstraintPtr& constraint) { constraint->dependencies.emplace_back(fcc); }); return InferencePack{rets, {refinementArena.variadic(returnRefinements)}}; } } Inference ConstraintGenerator::check( const ScopePtr& scope, AstExpr* expr, std::optional expectedType, bool forceSingleton, bool generalize) { RecursionCounter counter{&recursionCount}; if (recursionCount >= FInt::LuauCheckRecursionLimit) { reportCodeTooComplex(expr->location); return Inference{builtinTypes->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{builtinTypes->numberType}; else if (auto boolExpr = expr->as()) result = check(scope, boolExpr, expectedType, forceSingleton); else if (expr->is()) result = Inference{builtinTypes->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()) result = flattenPack(scope, expr->location, checkPack(scope, call)); // TODO: needs predicates too else if (auto a = expr->as()) result = check(scope, a, expectedType, generalize); 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 interpString = expr->as()) result = check(scope, interpString); else if (auto err = expr->as()) { // Open question: Should we traverse into this? for (AstExpr* subExpr : err->expressions) check(scope, subExpr); result = Inference{builtinTypes->errorRecoveryType()}; } else { LUAU_ASSERT(0); result = Inference{freshType(scope)}; } LUAU_ASSERT(result.ty); module->astTypes[expr] = result.ty; if (expectedType) module->astExpectedTypes[expr] = *expectedType; return result; } Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprConstantString* string, std::optional expectedType, bool forceSingleton) { if (forceSingleton) return Inference{arena->addType(SingletonType{StringSingleton{std::string{string->value.data, string->value.size}}})}; if (expectedType) { const TypeId expectedTy = follow(*expectedType); if (get(expectedTy) || get(expectedTy) || get(expectedTy)) { TypeId ty = arena->addType(BlockedType{}); TypeId singletonType = arena->addType(SingletonType(StringSingleton{std::string(string->value.data, string->value.size)})); addConstraint(scope, string->location, PrimitiveTypeConstraint{ty, expectedTy, singletonType, builtinTypes->stringType}); return Inference{ty}; } else if (maybeSingleton(expectedTy)) return Inference{arena->addType(SingletonType{StringSingleton{std::string{string->value.data, string->value.size}}})}; return Inference{builtinTypes->stringType}; } return Inference{builtinTypes->stringType}; } Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprConstantBool* boolExpr, std::optional expectedType, bool forceSingleton) { const TypeId singletonType = boolExpr->value ? builtinTypes->trueType : builtinTypes->falseType; if (forceSingleton) return Inference{singletonType}; if (expectedType) { const TypeId expectedTy = follow(*expectedType); if (get(expectedTy) || get(expectedTy)) { TypeId ty = arena->addType(BlockedType{}); addConstraint(scope, boolExpr->location, PrimitiveTypeConstraint{ty, expectedTy, singletonType, builtinTypes->booleanType}); return Inference{ty}; } else if (maybeSingleton(expectedTy)) return Inference{singletonType}; return Inference{builtinTypes->booleanType}; } return Inference{builtinTypes->booleanType}; } Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprLocal* local) { const RefinementKey* key = dfg->getRefinementKey(local); std::optional rvalueDef = dfg->getRValueDefForCompoundAssign(local); LUAU_ASSERT(key || rvalueDef); std::optional maybeTy; // if we have a refinement key, we can look up its type. if (key) maybeTy = lookup(scope.get(), key->def); // if the current def doesn't have a type, we might be doing a compound assignment // and therefore might need to look at the rvalue def instead. if (!maybeTy && rvalueDef) maybeTy = lookup(scope.get(), *rvalueDef); if (maybeTy) { TypeId ty = follow(*maybeTy); recordInferredBinding(local->local, ty); return Inference{ty, refinementArena.proposition(key, builtinTypes->truthyType)}; } else ice->ice("CG: AstExprLocal came before its declaration?"); } Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprGlobal* global) { const RefinementKey* key = dfg->getRefinementKey(global); std::optional rvalueDef = dfg->getRValueDefForCompoundAssign(global); LUAU_ASSERT(key || rvalueDef); // we'll use whichever of the two definitions we have here. DefId def = key ? key->def : *rvalueDef; /* 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. */ if (auto ty = lookup(scope.get(), def, /*prototype=*/false)) { rootScope->lvalueTypes[def] = *ty; return Inference{*ty, refinementArena.proposition(key, builtinTypes->truthyType)}; } else { reportError(global->location, UnknownSymbol{global->name.value, UnknownSymbol::Binding}); return Inference{builtinTypes->errorRecoveryType()}; } } Inference ConstraintGenerator::checkIndexName(const ScopePtr& scope, const RefinementKey* key, AstExpr* indexee, std::string index) { TypeId obj = check(scope, indexee).ty; TypeId result = arena->addType(BlockedType{}); if (key) { if (auto ty = lookup(scope.get(), key->def)) return Inference{*ty, refinementArena.proposition(key, builtinTypes->truthyType)}; scope->rvalueRefinements[key->def] = result; } addConstraint(scope, indexee->location, HasPropConstraint{result, obj, std::move(index)}); if (key) return Inference{result, refinementArena.proposition(key, builtinTypes->truthyType)}; else return Inference{result}; } Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprIndexName* indexName) { const RefinementKey* key = dfg->getRefinementKey(indexName); return checkIndexName(scope, key, indexName->expr, indexName->index.value); } Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprIndexExpr* indexExpr) { if (auto constantString = indexExpr->index->as()) { const RefinementKey* key = dfg->getRefinementKey(indexExpr); return checkIndexName(scope, key, indexExpr->expr, constantString->value.data); } TypeId obj = check(scope, indexExpr->expr).ty; TypeId indexType = check(scope, indexExpr->index).ty; TypeId result = freshType(scope); const RefinementKey* key = dfg->getRefinementKey(indexExpr); if (key) { if (auto ty = lookup(scope.get(), key->def)) return Inference{*ty, refinementArena.proposition(key, builtinTypes->truthyType)}; scope->rvalueRefinements[key->def] = result; } TableIndexer indexer{indexType, result}; TypeId tableType = arena->addType(TableType{TableType::Props{}, TableIndexer{indexType, result}, TypeLevel{}, scope.get(), TableState::Free}); addConstraint(scope, indexExpr->expr->location, SubtypeConstraint{obj, tableType}); if (key) return Inference{result, refinementArena.proposition(key, builtinTypes->truthyType)}; else return Inference{result}; } Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprFunction* func, std::optional expectedType, bool generalize) { Checkpoint startCheckpoint = checkpoint(this); FunctionSignature sig = checkFunctionSignature(scope, func, expectedType); checkFunctionBody(sig.bodyScope, func); Checkpoint endCheckpoint = checkpoint(this); if (generalize) { TypeId generalizedTy = arena->addType(BlockedType{}); NotNull gc = addConstraint(sig.signatureScope, func->location, GeneralizationConstraint{generalizedTy, sig.signature}); Constraint* previous = nullptr; forEachConstraint(startCheckpoint, endCheckpoint, this, [gc, &previous](const ConstraintPtr& constraint) { gc->dependencies.emplace_back(constraint.get()); if (auto psc = get(*constraint); psc && psc->returns) { if (previous) constraint->dependencies.push_back(NotNull{previous}); previous = constraint.get(); } }); return Inference{generalizedTy}; } else { return Inference{sig.signature}; } } Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprUnary* unary) { auto [operandType, refinement] = check(scope, unary->expr); switch (unary->op) { case AstExprUnary::Op::Not: { TypeId resultType = arena->addType(TypeFamilyInstanceType{ NotNull{&kBuiltinTypeFamilies.notFamily}, {operandType}, {}, }); addConstraint(scope, unary->location, ReduceConstraint{resultType}); return Inference{resultType, refinementArena.negation(refinement)}; } case AstExprUnary::Op::Len: { TypeId resultType = arena->addType(TypeFamilyInstanceType{ NotNull{&kBuiltinTypeFamilies.lenFamily}, {operandType}, {}, }); addConstraint(scope, unary->location, ReduceConstraint{resultType}); return Inference{resultType, refinementArena.negation(refinement)}; } case AstExprUnary::Op::Minus: { TypeId resultType = arena->addType(TypeFamilyInstanceType{ NotNull{&kBuiltinTypeFamilies.unmFamily}, {operandType}, {}, }); addConstraint(scope, unary->location, ReduceConstraint{resultType}); return Inference{resultType, refinementArena.negation(refinement)}; } default: // msvc can't prove that this is exhaustive. LUAU_UNREACHABLE(); } } Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprBinary* binary, std::optional expectedType) { auto [leftType, rightType, refinement] = checkBinary(scope, binary, expectedType); switch (binary->op) { case AstExprBinary::Op::Add: { TypeId resultType = arena->addType(TypeFamilyInstanceType{ NotNull{&kBuiltinTypeFamilies.addFamily}, {leftType, rightType}, {}, }); addConstraint(scope, binary->location, ReduceConstraint{resultType}); return Inference{resultType, std::move(refinement)}; } case AstExprBinary::Op::Sub: { TypeId resultType = arena->addType(TypeFamilyInstanceType{ NotNull{&kBuiltinTypeFamilies.subFamily}, {leftType, rightType}, {}, }); addConstraint(scope, binary->location, ReduceConstraint{resultType}); return Inference{resultType, std::move(refinement)}; } case AstExprBinary::Op::Mul: { TypeId resultType = arena->addType(TypeFamilyInstanceType{ NotNull{&kBuiltinTypeFamilies.mulFamily}, {leftType, rightType}, {}, }); addConstraint(scope, binary->location, ReduceConstraint{resultType}); return Inference{resultType, std::move(refinement)}; } case AstExprBinary::Op::Div: { TypeId resultType = arena->addType(TypeFamilyInstanceType{ NotNull{&kBuiltinTypeFamilies.divFamily}, {leftType, rightType}, {}, }); addConstraint(scope, binary->location, ReduceConstraint{resultType}); return Inference{resultType, std::move(refinement)}; } case AstExprBinary::Op::FloorDiv: { TypeId resultType = arena->addType(TypeFamilyInstanceType{ NotNull{&kBuiltinTypeFamilies.idivFamily}, {leftType, rightType}, {}, }); addConstraint(scope, binary->location, ReduceConstraint{resultType}); return Inference{resultType, std::move(refinement)}; } case AstExprBinary::Op::Pow: { TypeId resultType = arena->addType(TypeFamilyInstanceType{ NotNull{&kBuiltinTypeFamilies.powFamily}, {leftType, rightType}, {}, }); addConstraint(scope, binary->location, ReduceConstraint{resultType}); return Inference{resultType, std::move(refinement)}; } case AstExprBinary::Op::Mod: { TypeId resultType = arena->addType(TypeFamilyInstanceType{ NotNull{&kBuiltinTypeFamilies.modFamily}, {leftType, rightType}, {}, }); addConstraint(scope, binary->location, ReduceConstraint{resultType}); return Inference{resultType, std::move(refinement)}; } case AstExprBinary::Op::Concat: { TypeId resultType = arena->addType(TypeFamilyInstanceType{ NotNull{&kBuiltinTypeFamilies.concatFamily}, {leftType, rightType}, {}, }); addConstraint(scope, binary->location, ReduceConstraint{resultType}); return Inference{resultType, std::move(refinement)}; } case AstExprBinary::Op::And: { TypeId resultType = arena->addType(TypeFamilyInstanceType{ NotNull{&kBuiltinTypeFamilies.andFamily}, {leftType, rightType}, {}, }); addConstraint(scope, binary->location, ReduceConstraint{resultType}); return Inference{resultType, std::move(refinement)}; } case AstExprBinary::Op::Or: { TypeId resultType = arena->addType(TypeFamilyInstanceType{ NotNull{&kBuiltinTypeFamilies.orFamily}, {leftType, rightType}, {}, }); addConstraint(scope, binary->location, ReduceConstraint{resultType}); return Inference{resultType, std::move(refinement)}; } case AstExprBinary::Op::CompareLt: { TypeId resultType = arena->addType(TypeFamilyInstanceType{ NotNull{&kBuiltinTypeFamilies.ltFamily}, {leftType, rightType}, {}, }); addConstraint(scope, binary->location, ReduceConstraint{resultType}); return Inference{resultType, std::move(refinement)}; } case AstExprBinary::Op::CompareGe: { TypeId resultType = arena->addType(TypeFamilyInstanceType{ NotNull{&kBuiltinTypeFamilies.ltFamily}, {rightType, leftType}, // lua decided that `__ge(a, b)` is instead just `__lt(b, a)` {}, }); addConstraint(scope, binary->location, ReduceConstraint{resultType}); return Inference{resultType, std::move(refinement)}; } case AstExprBinary::Op::CompareLe: { TypeId resultType = arena->addType(TypeFamilyInstanceType{ NotNull{&kBuiltinTypeFamilies.leFamily}, {leftType, rightType}, {}, }); addConstraint(scope, binary->location, ReduceConstraint{resultType}); return Inference{resultType, std::move(refinement)}; } case AstExprBinary::Op::CompareGt: { TypeId resultType = arena->addType(TypeFamilyInstanceType{ NotNull{&kBuiltinTypeFamilies.leFamily}, {rightType, leftType}, // lua decided that `__gt(a, b)` is instead just `__le(b, a)` {}, }); addConstraint(scope, binary->location, ReduceConstraint{resultType}); return Inference{resultType, std::move(refinement)}; } case AstExprBinary::Op::CompareEq: case AstExprBinary::Op::CompareNe: { TypeId resultType = arena->addType(TypeFamilyInstanceType{ NotNull{&kBuiltinTypeFamilies.eqFamily}, {leftType, rightType}, {}, }); addConstraint(scope, binary->location, ReduceConstraint{resultType}); return Inference{resultType, std::move(refinement)}; } case AstExprBinary::Op::Op__Count: ice->ice("Op__Count should never be generated in an AST."); default: // msvc can't prove that this is exhaustive. LUAU_UNREACHABLE(); } } Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprIfElse* ifElse, std::optional expectedType) { ScopePtr condScope = childScope(ifElse->condition, scope); RefinementId refinement = check(condScope, ifElse->condition).refinement; ScopePtr thenScope = childScope(ifElse->trueExpr, scope); applyRefinements(thenScope, ifElse->trueExpr->location, refinement); TypeId thenType = check(thenScope, ifElse->trueExpr, expectedType).ty; ScopePtr elseScope = childScope(ifElse->falseExpr, scope); applyRefinements(elseScope, ifElse->falseExpr->location, refinementArena.negation(refinement)); TypeId elseType = check(elseScope, ifElse->falseExpr, expectedType).ty; return Inference{expectedType ? *expectedType : simplifyUnion(builtinTypes, arena, thenType, elseType).result}; } Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprTypeAssertion* typeAssert) { check(scope, typeAssert->expr, std::nullopt); return Inference{resolveType(scope, typeAssert->annotation, /* inTypeArguments */ false)}; } Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprInterpString* interpString) { for (AstExpr* expr : interpString->expressions) check(scope, expr); return Inference{builtinTypes->stringType}; } std::tuple ConstraintGenerator::checkBinary( const ScopePtr& scope, AstExprBinary* binary, std::optional expectedType) { if (binary->op == AstExprBinary::And) { std::optional relaxedExpectedLhs; if (expectedType) relaxedExpectedLhs = arena->addType(UnionType{{builtinTypes->falsyType, *expectedType}}); auto [leftType, leftRefinement] = check(scope, binary->left, relaxedExpectedLhs); ScopePtr rightScope = childScope(binary->right, scope); applyRefinements(rightScope, binary->right->location, leftRefinement); auto [rightType, rightRefinement] = check(rightScope, binary->right, expectedType); return {leftType, rightType, refinementArena.conjunction(leftRefinement, rightRefinement)}; } else if (binary->op == AstExprBinary::Or) { std::optional relaxedExpectedLhs; if (expectedType) relaxedExpectedLhs = arena->addType(UnionType{{builtinTypes->falsyType, *expectedType}}); auto [leftType, leftRefinement] = check(scope, binary->left, relaxedExpectedLhs); ScopePtr rightScope = childScope(binary->right, scope); applyRefinements(rightScope, binary->right->location, refinementArena.negation(leftRefinement)); auto [rightType, rightRefinement] = check(rightScope, binary->right, expectedType); return {leftType, rightType, refinementArena.disjunction(leftRefinement, rightRefinement)}; } else if (auto typeguard = matchTypeGuard(binary)) { TypeId leftType = check(scope, binary->left).ty; TypeId rightType = check(scope, binary->right).ty; const RefinementKey* key = dfg->getRefinementKey(typeguard->target); if (!key) return {leftType, rightType, nullptr}; auto augmentForErrorSupression = [&](TypeId ty) -> TypeId { return arena->addType(UnionType{{ty, builtinTypes->errorType}}); }; TypeId discriminantTy = builtinTypes->neverType; if (typeguard->type == "nil") discriminantTy = builtinTypes->nilType; else if (typeguard->type == "string") discriminantTy = builtinTypes->stringType; else if (typeguard->type == "number") discriminantTy = builtinTypes->numberType; else if (typeguard->type == "boolean") discriminantTy = builtinTypes->booleanType; else if (typeguard->type == "thread") discriminantTy = builtinTypes->threadType; else if (typeguard->type == "buffer") discriminantTy = builtinTypes->bufferType; else if (typeguard->type == "table") discriminantTy = augmentForErrorSupression(builtinTypes->tableType); else if (typeguard->type == "function") discriminantTy = augmentForErrorSupression(builtinTypes->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 = builtinTypes->classType; } else if (!typeguard->isTypeof && typeguard->type == "vector") discriminantTy = builtinTypes->neverType; // TODO: figure out a way to deal with this quirky type else if (!typeguard->isTypeof) discriminantTy = builtinTypes->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 == builtinTypes->classType) discriminantTy = ty; } RefinementId proposition = refinementArena.proposition(key, discriminantTy); if (binary->op == AstExprBinary::CompareEq) return {leftType, rightType, proposition}; else if (binary->op == AstExprBinary::CompareNe) return {leftType, rightType, refinementArena.negation(proposition)}; else ice->ice("matchTypeGuard should only return a Some under `==` or `~=`!"); } else if (binary->op == AstExprBinary::CompareEq || binary->op == AstExprBinary::CompareNe) { // We are checking a binary expression of the form a op b // Just because a op b is epxected to return a bool, doesn't mean a, b are expected to be bools too TypeId leftType = check(scope, binary->left, {}, true).ty; TypeId rightType = check(scope, binary->right, {}, true).ty; RefinementId leftRefinement = refinementArena.proposition(dfg->getRefinementKey(binary->left), rightType); RefinementId rightRefinement = refinementArena.proposition(dfg->getRefinementKey(binary->right), leftType); if (binary->op == AstExprBinary::CompareNe) { leftRefinement = refinementArena.negation(leftRefinement); rightRefinement = refinementArena.negation(rightRefinement); } return {leftType, rightType, refinementArena.equivalence(leftRefinement, rightRefinement)}; } else { TypeId leftType = check(scope, binary->left).ty; TypeId rightType = check(scope, binary->right).ty; return {leftType, rightType, nullptr}; } } std::optional ConstraintGenerator::checkLValue(const ScopePtr& scope, AstExpr* expr, TypeId assignedTy) { if (auto local = expr->as()) return checkLValue(scope, local, assignedTy); else if (auto global = expr->as()) return checkLValue(scope, global, assignedTy); else if (auto indexName = expr->as()) return checkLValue(scope, indexName, assignedTy); else if (auto indexExpr = expr->as()) return checkLValue(scope, indexExpr, assignedTy); else if (auto error = expr->as()) { check(scope, error); return builtinTypes->errorRecoveryType(); } else ice->ice("checkLValue is inexhaustive"); } std::optional ConstraintGenerator::checkLValue(const ScopePtr& scope, AstExprLocal* local, TypeId assignedTy) { std::optional annotatedTy = scope->lookup(local->local); LUAU_ASSERT(annotatedTy); if (annotatedTy) addConstraint(scope, local->location, SubtypeConstraint{assignedTy, *annotatedTy}); const DefId defId = dfg->getDef(local); std::optional ty = scope->lookupUnrefinedType(defId); if (ty) { if (auto lt = getMutable(*ty)) ++lt->blockCount; } else { ty = arena->addType(LocalType{builtinTypes->neverType, /* blockCount */ 1, local->local->name.value}); scope->lvalueTypes[defId] = *ty; } addConstraint(scope, local->location, UnpackConstraint{ arena->addTypePack({*ty}), arena->addTypePack({assignedTy}), /*resultIsLValue*/ true }); recordInferredBinding(local->local, *ty); return ty; } std::optional ConstraintGenerator::checkLValue(const ScopePtr& scope, AstExprGlobal* global, TypeId assignedTy) { return scope->lookup(Symbol{global->name}); } std::optional ConstraintGenerator::checkLValue(const ScopePtr& scope, AstExprIndexName* indexName, TypeId assignedTy) { return updateProperty(scope, indexName, assignedTy); } std::optional ConstraintGenerator::checkLValue(const ScopePtr& scope, AstExprIndexExpr* indexExpr, TypeId assignedTy) { return updateProperty(scope, indexExpr, assignedTy); } /** * This function is mostly about identifying properties that are being inserted into unsealed tables. * * If expr has the form name.a.b.c */ TypeId ConstraintGenerator::updateProperty(const ScopePtr& scope, AstExpr* expr, TypeId assignedTy) { // There are a bunch of cases where we realize that this is not the kind of // assignment that potentially changes the shape of a table. When we // encounter them, we call this to fall back and do the "usual thing." auto fallback = [&]() { TypeId resTy = check(scope, expr).ty; addConstraint(scope, expr->location, SubtypeConstraint{assignedTy, resTy}); return resTy; }; LUAU_ASSERT(expr->is() || expr->is()); if (auto indexExpr = expr->as(); indexExpr && !indexExpr->index->is()) { // An indexer is only interesting in an lvalue-ey way if it is at the // tail of an expression. // // If the indexer is not at the tail, then we are not interested in // augmenting the lhs data structure with a new indexer. Constraint // generation can treat it as an ordinary lvalue. // // eg // // a.b.c[1] = 44 -- lvalue // a.b[4].c = 2 -- rvalue TypeId resultType = arena->addType(BlockedType{}); TypeId subjectType = check(scope, indexExpr->expr).ty; TypeId indexType = check(scope, indexExpr->index).ty; addConstraint(scope, expr->location, SetIndexerConstraint{resultType, subjectType, indexType, assignedTy}); module->astTypes[expr] = assignedTy; return assignedTy; } Symbol sym; const Def* def = nullptr; std::vector segments; std::vector exprs; AstExpr* e = expr; while (e) { if (auto global = e->as()) { sym = global->name; def = dfg->getDef(global); break; } else if (auto local = e->as()) { sym = local->local; def = dfg->getDef(local); break; } else if (auto indexName = e->as()) { segments.push_back(indexName->index.value); exprs.push_back(e); e = indexName->expr; } else if (auto indexExpr = e->as()) { if (auto strIndex = indexExpr->index->as()) { // We need to populate astTypes for the index value. check(scope, indexExpr->index); segments.push_back(std::string(strIndex->value.data, strIndex->value.size)); exprs.push_back(e); e = indexExpr->expr; } else { return fallback(); } } else { return fallback(); } } LUAU_ASSERT(!segments.empty()); std::reverse(begin(segments), end(segments)); std::reverse(begin(exprs), end(exprs)); LUAU_ASSERT(def); std::optional> lookupResult = scope->lookupEx(NotNull{def}); if (!lookupResult) return fallback(); const auto [subjectType, subjectScope] = *lookupResult; std::vector segmentStrings(begin(segments), end(segments)); TypeId updatedType = arena->addType(BlockedType{}); addConstraint(scope, expr->location, SetPropConstraint{updatedType, subjectType, std::move(segmentStrings), assignedTy}); TypeId prevSegmentTy = updatedType; for (size_t i = 0; i < segments.size(); ++i) { TypeId segmentTy = arena->addType(BlockedType{}); module->astTypes[exprs[i]] = segmentTy; addConstraint(scope, expr->location, HasPropConstraint{segmentTy, prevSegmentTy, segments[i]}); prevSegmentTy = segmentTy; } module->astTypes[expr] = prevSegmentTy; module->astTypes[e] = updatedType; if (!subjectType->persistent) { subjectScope->bindings[sym].typeId = updatedType; // This can fail if the user is erroneously trying to augment a builtin // table like os or string. if (auto key = dfg->getRefinementKey(e)) { subjectScope->lvalueTypes[key->def] = updatedType; subjectScope->rvalueRefinements[key->def] = updatedType; } } return assignedTy; } Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprTable* expr, std::optional expectedType) { const bool expectedTypeIsFree = expectedType && get(follow(*expectedType)); TypeId ty = arena->addType(TableType{}); TableType* 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}); }; std::optional annotatedKeyType; std::optional annotatedIndexResultType; if (expectedType) { if (const TableType* ttv = get(follow(*expectedType))) { if (ttv->indexer) { annotatedKeyType.emplace(follow(ttv->indexer->indexType)); annotatedIndexResultType.emplace(ttv->indexer->indexResultType); } } } bool isIndexedResultType = false; std::optional pinnedIndexResultType; for (const AstExprTable::Item& item : expr->items) { std::optional expectedValueType; if (item.kind == AstExprTable::Item::Kind::General || item.kind == AstExprTable::Item::Kind::List) isIndexedResultType = true; if (item.key && expectedType && !expectedTypeIsFree) { if (auto stringKey = item.key->as()) { ErrorVec errorVec; std::optional propTy = findTablePropertyRespectingMeta(builtinTypes, errorVec, follow(*expectedType), stringKey->value.data, item.value->location); if (propTy) expectedValueType = propTy; else { expectedValueType = arena->addType(BlockedType{}); addConstraint(scope, item.value->location, HasPropConstraint{*expectedValueType, *expectedType, stringKey->value.data, /*suppressSimplification*/ true}); } } } // We'll resolve the expected index result type here with the following priority: // 1. Record table types - in which key, value pairs must be handled on a k,v pair basis. // In this case, the above if-statement will populate expectedValueType // 2. Someone places an annotation on a General or List table // Trust the annotation and have the solver inform them if they get it wrong // 3. Someone omits the annotation on a general or List table // Use the type of the first indexResultType as the expected type std::optional checkExpectedIndexResultType; if (expectedValueType) { checkExpectedIndexResultType = expectedValueType; } else if (annotatedIndexResultType) { checkExpectedIndexResultType = annotatedIndexResultType; } else if (pinnedIndexResultType) { checkExpectedIndexResultType = pinnedIndexResultType; } TypeId itemTy = check(scope, item.value, checkExpectedIndexResultType).ty; if (isIndexedResultType && !pinnedIndexResultType) pinnedIndexResultType = itemTy; 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, annotatedKeyType).ty; if (AstExprConstantString* key = item.key->as()) { ttv->props[key->value.begin()] = {itemTy}; } else { createIndexer(item.key->location, keyTy, itemTy); } } else { TypeId numberType = builtinTypes->numberType; // FIXME? The location isn't quite right here. Not sure what is // right. createIndexer(item.value->location, numberType, itemTy); } } return Inference{ty}; } ConstraintGenerator::FunctionSignature ConstraintGenerator::checkFunctionSignature( const ScopePtr& parent, AstExprFunction* fn, std::optional expectedType, std::optional originalName) { ScopePtr signatureScope = nullptr; ScopePtr bodyScope = nullptr; TypePackId returnType = nullptr; std::vector genericTypes; std::vector genericTypePacks; if (expectedType) expectedType = follow(*expectedType); bool hasGenerics = fn->generics.size > 0 || fn->genericPacks.size > 0; 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); if (hasGenerics) { 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); } for (const auto& [name, g] : genericPackDefinitions) { genericTypePacks.push_back(g.tp); } // Local variable works around an odd gcc 11.3 warning: may be used uninitialized std::optional none = std::nullopt; expectedType = none; } std::vector argTypes; std::vector> argNames; TypePack expectedArgPack; const FunctionType* expectedFunction = expectedType ? get(*expectedType) : nullptr; // This check ensures that expectedType is precisely optional and not any (since any is also an optional type) if (expectedType && isOptional(*expectedType) && !get(*expectedType)) { auto ut = get(*expectedType); for (auto u : ut) { if (get(u) && !isNil(u)) { expectedFunction = get(u); break; } } } if (expectedFunction) { expectedArgPack = extendTypePack(*arena, builtinTypes, expectedFunction->argTypes, fn->args.size); genericTypes = expectedFunction->generics; genericTypePacks = expectedFunction->genericPacks; } if (fn->self) { TypeId selfType = freshType(signatureScope); argTypes.push_back(selfType); argNames.emplace_back(FunctionArgument{fn->self->name.value, fn->self->location}); signatureScope->bindings[fn->self] = Binding{selfType, fn->self->location}; DefId def = dfg->getDef(fn->self); signatureScope->lvalueTypes[def] = selfType; signatureScope->rvalueRefinements[def] = selfType; } for (size_t i = 0; i < fn->args.size; ++i) { AstLocal* local = fn->args.data[i]; TypeId argTy = nullptr; if (local->annotation) argTy = resolveType(signatureScope, local->annotation, /* inTypeArguments */ false, /* replaceErrorWithFresh*/ true); else { if (i < expectedArgPack.head.size()) argTy = expectedArgPack.head[i]; else argTy = freshType(signatureScope); } argTypes.push_back(argTy); argNames.emplace_back(FunctionArgument{local->name.value, local->location}); signatureScope->bindings[local] = Binding{argTy, local->location}; DefId def = dfg->getDef(local); signatureScope->lvalueTypes[def] = argTy; signatureScope->rvalueRefinements[def] = argTy; } TypePackId varargPack = nullptr; if (fn->vararg) { if (fn->varargAnnotation) { TypePackId annotationType = resolveTypePack(signatureScope, fn->varargAnnotation, /* inTypeArguments */ false, /* replaceErrorWithFresh */ true); varargPack = annotationType; } else if (expectedArgPack.tail && get(*expectedArgPack.tail)) varargPack = *expectedArgPack.tail; else varargPack = builtinTypes->anyTypePack; signatureScope->varargPack = varargPack; bodyScope->varargPack = varargPack; } else { varargPack = arena->addTypePack(VariadicTypePack{builtinTypes->anyType, /*hidden*/ true}); // We do not add to signatureScope->varargPack because ... is not valid // in functions without an explicit ellipsis. signatureScope->varargPack = std::nullopt; bodyScope->varargPack = std::nullopt; } LUAU_ASSERT(nullptr != varargPack); // If there is both an annotation and an expected type, the annotation wins. // Type checking will sort out any discrepancies later. if (fn->returnAnnotation) { TypePackId annotatedRetType = resolveTypePack(signatureScope, *fn->returnAnnotation, /* inTypeArguments */ false, /* replaceErrorWithFresh*/ true); // 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); } else if (expectedFunction) { asMutable(returnType)->ty.emplace(expectedFunction->retTypes); } // TODO: Preserve argument names in the function's type. FunctionType actualFunction{TypeLevel{}, parent.get(), arena->addTypePack(argTypes, varargPack), returnType}; actualFunction.generics = std::move(genericTypes); actualFunction.genericPacks = std::move(genericTypePacks); actualFunction.argNames = std::move(argNames); actualFunction.hasSelf = fn->self != nullptr; FunctionDefinition defn; defn.definitionModuleName = module->name; defn.definitionLocation = fn->location; defn.varargLocation = fn->vararg ? std::make_optional(fn->varargLocation) : std::nullopt; defn.originalNameLocation = originalName.value_or(Location(fn->location.begin, 0)); actualFunction.definition = defn; TypeId actualFunctionType = arena->addType(std::move(actualFunction)); LUAU_ASSERT(actualFunctionType); module->astTypes[fn] = actualFunctionType; if (expectedType && get(*expectedType)) bindFreeType(*expectedType, actualFunctionType); return { /* signature */ actualFunctionType, /* signatureScope */ signatureScope, /* bodyScope */ bodyScope, }; } void ConstraintGenerator::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 CG retain one of these forever addConstraint(scope, fn->location, PackSubtypeConstraint{scope->returnType, empty}); } } TypeId ConstraintGenerator::resolveType(const ScopePtr& scope, AstType* ty, bool inTypeArguments, bool replaceErrorWithFresh) { 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"}); module->astResolvedTypes[ty] = builtinTypes->errorRecoveryType(); return builtinTypes->errorRecoveryType(); } else return resolveType(scope, ref->parameters.data[0].type, inTypeArguments); } } 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, /* inTypeArguments */ true)); } else if (p.typePack) { packParameters.push_back(resolveTypePack(scope, p.typePack, /* inTypeArguments */ true)); } else { // This indicates a parser bug: one of these two pointers // should be set. LUAU_ASSERT(false); } } result = arena->addType(PendingExpansionType{ref->prefix, ref->name, parameters, packParameters}); // If we're not in a type argument context, we need to create a constraint that expands this. // The dispatching of the above constraint will queue up additional constraints for nested // type function applications. if (!inTypeArguments) addConstraint(scope, ty->location, TypeAliasExpansionConstraint{/* target */ result}); } } else { result = builtinTypes->errorRecoveryType(); if (replaceErrorWithFresh) result = freshType(scope); } } else if (auto tab = ty->as()) { TableType::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, inTypeArguments); // TODO: Fill in location. props[name] = {propTy}; } if (tab->indexer) { // TODO: Recursion limit. indexer = TableIndexer{ resolveType(scope, tab->indexer->indexType, inTypeArguments), resolveType(scope, tab->indexer->resultType, inTypeArguments), }; } result = arena->addType(TableType{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); } for (const auto& [name, g] : genericPackDefinitions) { genericTypePacks.push_back(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, inTypeArguments, replaceErrorWithFresh); TypePackId returnTypes = resolveTypePack(signatureScope, fn->returnTypes, inTypeArguments, replaceErrorWithFresh); // TODO: FunctionType needs a pointer to the scope so that we know // how to quantify/instantiate it. FunctionType ftv{TypeLevel{}, scope.get(), {}, {}, argTypes, returnTypes}; ftv.isCheckedFunction = fn->checkedFunction; // This replicates the behavior of the appropriate FunctionType // constructors. 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, inTypeArguments)); } result = arena->addType(UnionType{parts}); } else if (auto intersectionAnnotation = ty->as()) { std::vector parts; for (AstType* part : intersectionAnnotation->types) { // TODO: Recursion limit. parts.push_back(resolveType(scope, part, inTypeArguments)); } result = arena->addType(IntersectionType{parts}); } else if (auto boolAnnotation = ty->as()) { if (boolAnnotation->value) result = builtinTypes->trueType; else result = builtinTypes->falseType; } else if (auto stringAnnotation = ty->as()) { result = arena->addType(SingletonType(StringSingleton{std::string(stringAnnotation->value.data, stringAnnotation->value.size)})); } else if (ty->is()) { result = builtinTypes->errorRecoveryType(); if (replaceErrorWithFresh) result = freshType(scope); } else { LUAU_ASSERT(0); result = builtinTypes->errorRecoveryType(); } module->astResolvedTypes[ty] = result; return result; } TypePackId ConstraintGenerator::resolveTypePack(const ScopePtr& scope, AstTypePack* tp, bool inTypeArgument, bool replaceErrorWithFresh) { TypePackId result; if (auto expl = tp->as()) { result = resolveTypePack(scope, expl->typeList, inTypeArgument, replaceErrorWithFresh); } else if (auto var = tp->as()) { TypeId ty = resolveType(scope, var->variadicType, inTypeArgument, replaceErrorWithFresh); 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 = builtinTypes->errorRecoveryTypePack(); } } else { LUAU_ASSERT(0); result = builtinTypes->errorRecoveryTypePack(); } module->astResolvedTypePacks[tp] = result; return result; } TypePackId ConstraintGenerator::resolveTypePack(const ScopePtr& scope, const AstTypeList& list, bool inTypeArguments, bool replaceErrorWithFresh) { std::vector head; for (AstType* headTy : list.types) { head.push_back(resolveType(scope, headTy, inTypeArguments, replaceErrorWithFresh)); } std::optional tail = std::nullopt; if (list.tailType) { tail = resolveTypePack(scope, list.tailType, inTypeArguments, replaceErrorWithFresh); } return arena->addTypePack(TypePack{head, tail}); } std::vector> ConstraintGenerator::createGenerics( const ScopePtr& scope, AstArray generics, bool useCache, bool addTypes) { std::vector> result; for (const auto& generic : generics) { TypeId genericTy = nullptr; if (auto it = scope->parent->typeAliasTypeParameters.find(generic.name.value); useCache && it != scope->parent->typeAliasTypeParameters.end()) genericTy = it->second; else { genericTy = arena->addType(GenericType{scope.get(), generic.name.value}); scope->parent->typeAliasTypeParameters[generic.name.value] = genericTy; } std::optional defaultTy = std::nullopt; if (generic.defaultValue) defaultTy = resolveType(scope, generic.defaultValue, /* inTypeArguments */ false); if (addTypes) scope->privateTypeBindings[generic.name.value] = TypeFun{genericTy}; result.push_back({generic.name.value, GenericTypeDefinition{genericTy, defaultTy}}); } return result; } std::vector> ConstraintGenerator::createGenericPacks( const ScopePtr& scope, AstArray generics, bool useCache, bool addTypes) { std::vector> result; for (const auto& generic : generics) { TypePackId genericTy; if (auto it = scope->parent->typeAliasTypePackParameters.find(generic.name.value); useCache && it != scope->parent->typeAliasTypePackParameters.end()) genericTy = it->second; else { genericTy = arena->addTypePack(TypePackVar{GenericTypePack{scope.get(), generic.name.value}}); scope->parent->typeAliasTypePackParameters[generic.name.value] = genericTy; } std::optional defaultTy = std::nullopt; if (generic.defaultValue) defaultTy = resolveTypePack(scope, generic.defaultValue, /* inTypeArguments */ false); if (addTypes) scope->privateTypePackBindings[generic.name.value] = genericTy; result.push_back({generic.name.value, GenericTypePackDefinition{genericTy, defaultTy}}); } return result; } Inference ConstraintGenerator::flattenPack(const ScopePtr& scope, Location location, InferencePack pack) { const auto& [tp, refinements] = pack; RefinementId refinement = nullptr; if (!refinements.empty()) refinement = refinements[0]; if (auto f = first(tp)) return Inference{*f, refinement}; TypeId typeResult = arena->addType(BlockedType{}); TypePackId resultPack = arena->addTypePack({typeResult}, arena->freshTypePack(scope.get())); addConstraint(scope, location, UnpackConstraint{resultPack, tp}); return Inference{typeResult, refinement}; } void ConstraintGenerator::reportError(Location location, TypeErrorData err) { errors.push_back(TypeError{location, module->name, std::move(err)}); if (logger) logger->captureGenerationError(errors.back()); } void ConstraintGenerator::reportCodeTooComplex(Location location) { errors.push_back(TypeError{location, module->name, CodeTooComplex{}}); if (logger) logger->captureGenerationError(errors.back()); } struct GlobalPrepopulator : AstVisitor { const NotNull globalScope; const NotNull arena; const NotNull dfg; GlobalPrepopulator(NotNull globalScope, NotNull arena, NotNull dfg) : globalScope(globalScope) , arena(arena) , dfg(dfg) { } bool visit(AstExprGlobal* global) override { if (auto ty = globalScope->lookup(global->name)) { DefId def = dfg->getDef(global); globalScope->lvalueTypes[def] = *ty; } return true; } bool visit(AstStatFunction* function) override { if (AstExprGlobal* g = function->name->as()) { TypeId bt = arena->addType(BlockedType{}); globalScope->bindings[g->name] = Binding{bt}; } return true; } bool visit(AstType*) override { return true; } bool visit(class AstTypePack* node) override { return true; } }; void ConstraintGenerator::prepopulateGlobalScope(const ScopePtr& globalScope, AstStatBlock* program) { GlobalPrepopulator gp{NotNull{globalScope.get()}, arena, dfg}; if (prepareModuleScope) prepareModuleScope(module->name, globalScope); program->visit(&gp); } void ConstraintGenerator::recordInferredBinding(AstLocal* local, TypeId ty) { if (InferredBinding* ib = inferredBindings.find(local)) ib->types.insert(ty); } void ConstraintGenerator::fillInInferredBindings(const ScopePtr& globalScope, AstStatBlock* block) { for (const auto& [symbol, p] : inferredBindings) { const auto& [scope, location, types] = p; std::vector tys(types.begin(), types.end()); TypeId ty = arena->addType(BlockedType{}); addConstraint(globalScope, Location{}, SetOpConstraint{SetOpConstraint::Union, ty, std::move(tys)}); scope->bindings[symbol] = Binding{ty, location}; } } std::vector> ConstraintGenerator::getExpectedCallTypesForFunctionOverloads(const TypeId fnType) { std::vector funTys; if (auto it = get(follow(fnType))) { for (TypeId intersectionComponent : it) { funTys.push_back(intersectionComponent); } } std::vector> expectedTypes; // For a list of functions f_0 : e_0 -> r_0, ... f_n : e_n -> r_n, // emit a list of arguments that the function could take at each position // by unioning the arguments at each place auto assignOption = [this, &expectedTypes](size_t index, TypeId ty) { if (index == expectedTypes.size()) expectedTypes.push_back(ty); else if (ty) { auto& el = expectedTypes[index]; if (!el) el = ty; else { std::vector result = reduceUnion({*el, ty}); if (result.empty()) el = builtinTypes->neverType; else if (result.size() == 1) el = result[0]; else el = module->internalTypes.addType(UnionType{std::move(result)}); } } }; for (const TypeId overload : funTys) { if (const FunctionType* ftv = get(follow(overload))) { auto [argsHead, argsTail] = flatten(ftv->argTypes); size_t start = ftv->hasSelf ? 1 : 0; size_t index = 0; for (size_t i = start; i < argsHead.size(); ++i) assignOption(index++, argsHead[i]); if (argsTail) { argsTail = follow(*argsTail); if (const VariadicTypePack* vtp = get(*argsTail)) { while (index < funTys.size()) assignOption(index++, vtp->ty); } } } } // TODO vvijay Feb 24, 2023 apparently we have to demote the types here? return expectedTypes; } 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