2022-06-03 21:32:20 +01:00
|
|
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
2023-11-03 19:47:28 +00:00
|
|
|
#include "Luau/ConstraintGenerator.h"
|
2022-10-21 18:33:43 +01:00
|
|
|
|
2022-08-11 21:42:54 +01:00
|
|
|
#include "Luau/Ast.h"
|
2023-10-20 21:36:26 +01:00
|
|
|
#include "Luau/Def.h"
|
2022-08-18 22:04:33 +01:00
|
|
|
#include "Luau/Common.h"
|
2022-08-11 21:42:54 +01:00
|
|
|
#include "Luau/Constraint.h"
|
2023-03-17 14:59:30 +00:00
|
|
|
#include "Luau/ControlFlow.h"
|
2022-10-21 18:33:43 +01:00
|
|
|
#include "Luau/DcrLogger.h"
|
2023-11-10 18:05:48 +00:00
|
|
|
#include "Luau/DenseHash.h"
|
2022-09-02 00:00:14 +01:00
|
|
|
#include "Luau/ModuleResolver.h"
|
2022-07-01 00:29:02 +01:00
|
|
|
#include "Luau/RecursionCounter.h"
|
2023-03-03 13:45:38 +00:00
|
|
|
#include "Luau/Refinement.h"
|
2022-10-21 18:33:43 +01:00
|
|
|
#include "Luau/Scope.h"
|
|
|
|
#include "Luau/TypeUtils.h"
|
2023-01-03 17:33:19 +00:00
|
|
|
#include "Luau/Type.h"
|
2023-05-19 19:59:59 +01:00
|
|
|
#include "Luau/TypeFamily.h"
|
|
|
|
#include "Luau/Simplify.h"
|
|
|
|
#include "Luau/VisitType.h"
|
2023-06-09 13:20:36 +01:00
|
|
|
#include "Luau/InsertionOrderedMap.h"
|
2022-07-01 00:29:02 +01:00
|
|
|
|
2023-02-17 14:53:37 +00:00
|
|
|
#include <algorithm>
|
2024-01-27 02:30:40 +00:00
|
|
|
#include <memory>
|
2023-02-17 14:53:37 +00:00
|
|
|
|
2022-07-01 00:29:02 +01:00
|
|
|
LUAU_FASTINT(LuauCheckRecursionLimit);
|
2023-08-04 18:01:35 +01:00
|
|
|
LUAU_FASTFLAG(DebugLuauLogSolverToJson);
|
2022-09-29 23:11:54 +01:00
|
|
|
LUAU_FASTFLAG(DebugLuauMagicTypes);
|
2023-09-30 01:22:06 +01:00
|
|
|
LUAU_FASTFLAG(LuauLoopControlFlowAnalysis);
|
2022-06-03 21:32:20 +01:00
|
|
|
|
|
|
|
namespace Luau
|
|
|
|
{
|
|
|
|
|
2023-03-24 17:34:14 +00:00
|
|
|
bool doesCallError(const AstExprCall* call); // TypeInfer.cpp
|
2022-06-17 01:54:42 +01:00
|
|
|
const AstStat* getFallthrough(const AstStat* node); // TypeInfer.cpp
|
2022-06-03 21:32:20 +01:00
|
|
|
|
2022-09-02 00:00:14 +01:00
|
|
|
static std::optional<AstExpr*> matchRequire(const AstExprCall& call)
|
|
|
|
{
|
|
|
|
const char* require = "require";
|
|
|
|
|
|
|
|
if (call.args.size != 1)
|
|
|
|
return std::nullopt;
|
|
|
|
|
|
|
|
const AstExprGlobal* funcAsGlobal = call.func->as<AstExprGlobal>();
|
|
|
|
if (!funcAsGlobal || funcAsGlobal->name != require)
|
|
|
|
return std::nullopt;
|
|
|
|
|
|
|
|
if (call.args.size != 1)
|
|
|
|
return std::nullopt;
|
|
|
|
|
|
|
|
return call.args.data[0];
|
|
|
|
}
|
|
|
|
|
2022-09-23 19:32:10 +01:00
|
|
|
static bool matchSetmetatable(const AstExprCall& call)
|
|
|
|
{
|
|
|
|
const char* smt = "setmetatable";
|
|
|
|
|
|
|
|
if (call.args.size != 2)
|
|
|
|
return false;
|
2022-10-13 23:59:53 +01:00
|
|
|
|
2022-09-23 19:32:10 +01:00
|
|
|
const AstExprGlobal* funcAsGlobal = call.func->as<AstExprGlobal>();
|
|
|
|
if (!funcAsGlobal || funcAsGlobal->name != smt)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-11-10 22:04:44 +00:00
|
|
|
struct TypeGuard
|
|
|
|
{
|
|
|
|
bool isTypeof;
|
|
|
|
AstExpr* target;
|
|
|
|
std::string type;
|
|
|
|
};
|
|
|
|
|
|
|
|
static std::optional<TypeGuard> 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<AstExprCall>())
|
|
|
|
std::swap(left, right);
|
|
|
|
|
|
|
|
if (!right->is<AstExprConstantString>())
|
|
|
|
return std::nullopt;
|
|
|
|
|
|
|
|
AstExprCall* call = left->as<AstExprCall>();
|
|
|
|
AstExprConstantString* string = right->as<AstExprConstantString>();
|
|
|
|
if (!call || !string)
|
|
|
|
return std::nullopt;
|
|
|
|
|
|
|
|
AstExprGlobal* callee = call->func->as<AstExprGlobal>();
|
|
|
|
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),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2023-02-10 18:50:54 +00:00
|
|
|
static bool matchAssert(const AstExprCall& call)
|
|
|
|
{
|
|
|
|
if (call.args.size < 1)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const AstExprGlobal* funcAsGlobal = call.func->as<AstExprGlobal>();
|
|
|
|
if (!funcAsGlobal || funcAsGlobal->name != "assert")
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-11-10 22:04:44 +00:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
|
|
|
|
struct Checkpoint
|
|
|
|
{
|
|
|
|
size_t offset;
|
|
|
|
};
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
Checkpoint checkpoint(const ConstraintGenerator* cg)
|
2022-11-10 22:04:44 +00:00
|
|
|
{
|
2023-11-03 19:47:28 +00:00
|
|
|
return Checkpoint{cg->constraints.size()};
|
2022-11-10 22:04:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename F>
|
2023-11-03 19:47:28 +00:00
|
|
|
void forEachConstraint(const Checkpoint& start, const Checkpoint& end, const ConstraintGenerator* cg, F f)
|
2022-11-10 22:04:44 +00:00
|
|
|
{
|
|
|
|
for (size_t i = start.offset; i < end.offset; ++i)
|
2023-11-03 19:47:28 +00:00
|
|
|
f(cg->constraints[i]);
|
2022-11-10 22:04:44 +00:00
|
|
|
}
|
|
|
|
|
2024-02-09 17:32:52 +00:00
|
|
|
struct HasFreeType : TypeOnceVisitor
|
|
|
|
{
|
|
|
|
bool result = false;
|
|
|
|
|
2024-02-16 01:25:31 +00:00
|
|
|
HasFreeType() {}
|
2024-02-09 17:32:52 +00:00
|
|
|
|
|
|
|
bool visit(TypeId ty) override
|
|
|
|
{
|
|
|
|
if (result || ty->persistent)
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(TypePackId tp) override
|
|
|
|
{
|
|
|
|
if (result)
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(TypeId ty, const ClassType&) override
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(TypeId ty, const FreeType&) override
|
|
|
|
{
|
|
|
|
result = true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(TypePackId ty, const FreeTypePack&) override
|
|
|
|
{
|
|
|
|
result = true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
bool hasFreeType(TypeId ty)
|
|
|
|
{
|
|
|
|
HasFreeType hft{};
|
|
|
|
hft.traverse(ty);
|
|
|
|
return hft.result;
|
|
|
|
}
|
|
|
|
|
2022-11-10 22:04:44 +00:00
|
|
|
} // namespace
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ConstraintGenerator::ConstraintGenerator(ModulePtr module, NotNull<Normalizer> normalizer, NotNull<ModuleResolver> moduleResolver,
|
2023-04-28 12:55:55 +01:00
|
|
|
NotNull<BuiltinTypes> builtinTypes, NotNull<InternalErrorReporter> ice, const ScopePtr& globalScope,
|
2023-07-28 12:37:00 +01:00
|
|
|
std::function<void(const ModuleName&, const ScopePtr&)> prepareModuleScope, DcrLogger* logger, NotNull<DataFlowGraph> dfg,
|
|
|
|
std::vector<RequireCycle> requireCycles)
|
2023-04-21 22:41:03 +01:00
|
|
|
: module(module)
|
2023-01-03 17:33:19 +00:00
|
|
|
, builtinTypes(builtinTypes)
|
2023-08-04 18:01:35 +01:00
|
|
|
, arena(normalizer->arena)
|
2022-06-03 21:32:20 +01:00
|
|
|
, rootScope(nullptr)
|
2022-10-21 18:33:43 +01:00
|
|
|
, dfg(dfg)
|
2023-08-04 18:01:35 +01:00
|
|
|
, normalizer(normalizer)
|
2022-09-02 00:00:14 +01:00
|
|
|
, moduleResolver(moduleResolver)
|
2022-07-01 00:29:02 +01:00
|
|
|
, ice(ice)
|
|
|
|
, globalScope(globalScope)
|
2023-04-28 12:55:55 +01:00
|
|
|
, prepareModuleScope(std::move(prepareModuleScope))
|
2023-07-28 12:37:00 +01:00
|
|
|
, requireCycles(std::move(requireCycles))
|
2022-09-08 22:44:50 +01:00
|
|
|
, logger(logger)
|
2022-06-03 21:32:20 +01:00
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
LUAU_ASSERT(module);
|
2022-06-03 21:32:20 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
void ConstraintGenerator::visitModuleRoot(AstStatBlock* block)
|
2023-09-30 01:22:06 +01:00
|
|
|
{
|
|
|
|
LUAU_ASSERT(scopes.empty());
|
|
|
|
LUAU_ASSERT(rootScope == nullptr);
|
|
|
|
ScopePtr scope = std::make_shared<Scope>(globalScope);
|
|
|
|
rootScope = scope.get();
|
|
|
|
scopes.emplace_back(block->location, scope);
|
2024-02-02 18:20:03 +00:00
|
|
|
rootScope->location = block->location;
|
2023-09-30 01:22:06 +01:00
|
|
|
module->astScopes[block] = NotNull{scope.get()};
|
|
|
|
|
|
|
|
rootScope->returnType = freshTypePack(scope);
|
|
|
|
|
2024-02-23 18:40:00 +00:00
|
|
|
TypeId moduleFnTy = arena->addType(FunctionType{TypeLevel{}, rootScope, builtinTypes->anyTypePack, rootScope->returnType});
|
|
|
|
interiorTypes.emplace_back();
|
|
|
|
|
2023-09-30 01:22:06 +01:00
|
|
|
prepopulateGlobalScope(scope, block);
|
|
|
|
|
2024-02-23 18:40:00 +00:00
|
|
|
Checkpoint start = checkpoint(this);
|
|
|
|
|
|
|
|
ControlFlow cf = visitBlockWithoutChildScope(scope, block);
|
|
|
|
if (cf == ControlFlow::None)
|
|
|
|
addConstraint(scope, block->location, PackSubtypeConstraint{builtinTypes->emptyTypePack, rootScope->returnType});
|
|
|
|
|
|
|
|
Checkpoint end = checkpoint(this);
|
|
|
|
|
|
|
|
NotNull<Constraint> genConstraint = addConstraint(scope, block->location, GeneralizationConstraint{arena->addType(BlockedType{}), moduleFnTy, std::move(interiorTypes.back())});
|
|
|
|
forEachConstraint(start, end, this, [genConstraint](const ConstraintPtr& c) {
|
|
|
|
genConstraint->dependencies.push_back(NotNull{c.get()});
|
|
|
|
});
|
|
|
|
|
|
|
|
interiorTypes.pop_back();
|
2023-09-30 01:22:06 +01:00
|
|
|
|
2023-10-13 20:38:31 +01:00
|
|
|
fillInInferredBindings(scope, block);
|
|
|
|
|
2023-09-30 01:22:06 +01:00
|
|
|
if (logger)
|
|
|
|
logger->captureGenerationModule(module);
|
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
TypeId ConstraintGenerator::freshType(const ScopePtr& scope)
|
2022-06-03 21:32:20 +01:00
|
|
|
{
|
2023-08-04 18:01:35 +01:00
|
|
|
return Luau::freshType(arena, builtinTypes, scope.get());
|
2022-06-03 21:32:20 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
TypePackId ConstraintGenerator::freshTypePack(const ScopePtr& scope)
|
2022-06-03 21:32:20 +01:00
|
|
|
{
|
2022-07-29 04:41:13 +01:00
|
|
|
FreeTypePack f{scope.get()};
|
2022-06-03 21:32:20 +01:00
|
|
|
return arena->addTypePack(TypePackVar{std::move(f)});
|
|
|
|
}
|
|
|
|
|
2024-02-02 18:20:03 +00:00
|
|
|
TypePackId ConstraintGenerator::addTypePack(std::vector<TypeId> head, std::optional<TypePackId> tail)
|
|
|
|
{
|
|
|
|
if (head.empty())
|
|
|
|
{
|
|
|
|
if (tail)
|
|
|
|
return *tail;
|
|
|
|
else
|
|
|
|
return builtinTypes->emptyTypePack;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return arena->addTypePack(TypePack{std::move(head), tail});
|
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ScopePtr ConstraintGenerator::childScope(AstNode* node, const ScopePtr& parent)
|
2022-06-03 21:32:20 +01:00
|
|
|
{
|
2022-07-29 04:41:13 +01:00
|
|
|
auto scope = std::make_shared<Scope>(parent);
|
2022-08-18 22:04:33 +01:00
|
|
|
scopes.emplace_back(node->location, scope);
|
2024-02-02 18:20:03 +00:00
|
|
|
scope->location = node->location;
|
2022-06-03 21:32:20 +01:00
|
|
|
|
2022-07-29 04:41:13 +01:00
|
|
|
scope->returnType = parent->returnType;
|
2022-09-08 22:44:50 +01:00
|
|
|
scope->varargPack = parent->varargPack;
|
2022-08-18 22:04:33 +01:00
|
|
|
|
|
|
|
parent->children.push_back(NotNull{scope.get()});
|
|
|
|
module->astScopes[node] = scope.get();
|
2022-06-03 21:32:20 +01:00
|
|
|
|
2022-07-29 04:41:13 +01:00
|
|
|
return scope;
|
2022-06-03 21:32:20 +01:00
|
|
|
}
|
|
|
|
|
2024-01-27 02:30:40 +00:00
|
|
|
std::optional<TypeId> ConstraintGenerator::lookup(const ScopePtr& scope, DefId def, bool prototype)
|
2023-11-10 18:05:48 +00:00
|
|
|
{
|
|
|
|
if (get<Cell>(def))
|
|
|
|
return scope->lookup(def);
|
|
|
|
if (auto phi = get<Phi>(def))
|
|
|
|
{
|
|
|
|
if (auto found = scope->lookup(def))
|
|
|
|
return *found;
|
2024-02-09 17:32:52 +00:00
|
|
|
else if (phi->operands.size() == 1)
|
|
|
|
return lookup(scope, phi->operands[0], prototype);
|
2023-12-08 15:42:54 +00:00
|
|
|
else if (!prototype)
|
|
|
|
return std::nullopt;
|
2023-11-10 18:05:48 +00:00
|
|
|
|
|
|
|
TypeId res = builtinTypes->neverType;
|
|
|
|
|
2023-11-17 18:15:31 +00:00
|
|
|
for (DefId operand : phi->operands)
|
2023-11-10 18:05:48 +00:00
|
|
|
{
|
2023-12-08 15:42:54 +00:00
|
|
|
// `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.
|
2024-02-16 01:25:31 +00:00
|
|
|
std::optional<TypeId> ty = lookup(scope, operand, /*prototype*/ false);
|
2023-12-08 15:42:54 +00:00
|
|
|
if (!ty)
|
|
|
|
{
|
|
|
|
ty = arena->addType(BlockedType{});
|
|
|
|
rootScope->lvalueTypes[operand] = *ty;
|
|
|
|
}
|
|
|
|
|
2024-01-27 02:30:40 +00:00
|
|
|
res = makeUnion(scope, Location{} /* TODO: can we provide a real location here? */, res, *ty);
|
2023-11-10 18:05:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
scope->lvalueTypes[def] = res;
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
ice->ice("ConstraintGenerator::lookup is inexhaustive?");
|
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
NotNull<Constraint> ConstraintGenerator::addConstraint(const ScopePtr& scope, const Location& location, ConstraintV cv)
|
2022-06-03 21:32:20 +01:00
|
|
|
{
|
2022-11-10 22:04:44 +00:00
|
|
|
return NotNull{constraints.emplace_back(new Constraint{NotNull{scope.get()}, location, std::move(cv)}).get()};
|
2022-06-03 21:32:20 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
NotNull<Constraint> ConstraintGenerator::addConstraint(const ScopePtr& scope, std::unique_ptr<Constraint> c)
|
2022-06-03 21:32:20 +01:00
|
|
|
{
|
2022-11-10 22:04:44 +00:00
|
|
|
return NotNull{constraints.emplace_back(std::move(c)).get()};
|
2022-06-03 21:32:20 +01:00
|
|
|
}
|
|
|
|
|
2024-02-16 01:25:31 +00:00
|
|
|
void ConstraintGenerator::unionRefinements(const ScopePtr& scope, Location location, const RefinementContext& lhs, const RefinementContext& rhs,
|
|
|
|
RefinementContext& dest, std::vector<ConstraintV>* constraints)
|
2022-11-04 17:02:37 +00:00
|
|
|
{
|
2023-05-19 19:59:59 +01:00
|
|
|
const auto intersect = [&](const std::vector<TypeId>& types) {
|
|
|
|
if (1 == types.size())
|
|
|
|
return types[0];
|
|
|
|
else if (2 == types.size())
|
2024-01-27 02:30:40 +00:00
|
|
|
return makeIntersect(scope, location, types[0], types[1]);
|
2023-05-19 19:59:59 +01:00
|
|
|
|
|
|
|
return arena->addType(IntersectionType{types});
|
|
|
|
};
|
|
|
|
|
2023-03-03 13:45:38 +00:00
|
|
|
for (auto& [def, partition] : lhs)
|
2022-11-04 17:02:37 +00:00
|
|
|
{
|
|
|
|
auto rhsIt = rhs.find(def);
|
|
|
|
if (rhsIt == rhs.end())
|
|
|
|
continue;
|
|
|
|
|
2023-03-03 13:45:38 +00:00
|
|
|
LUAU_ASSERT(!partition.discriminantTypes.empty());
|
|
|
|
LUAU_ASSERT(!rhsIt->second.discriminantTypes.empty());
|
2022-11-04 17:02:37 +00:00
|
|
|
|
2023-05-19 19:59:59 +01:00
|
|
|
TypeId leftDiscriminantTy = partition.discriminantTypes.size() == 1 ? partition.discriminantTypes[0] : intersect(partition.discriminantTypes);
|
2022-11-04 17:02:37 +00:00
|
|
|
|
2023-05-19 19:59:59 +01:00
|
|
|
TypeId rightDiscriminantTy =
|
|
|
|
rhsIt->second.discriminantTypes.size() == 1 ? rhsIt->second.discriminantTypes[0] : intersect(rhsIt->second.discriminantTypes);
|
2023-03-03 13:45:38 +00:00
|
|
|
|
2023-06-09 13:20:36 +01:00
|
|
|
dest.insert(def, {});
|
2024-01-27 02:30:40 +00:00
|
|
|
dest.get(def)->discriminantTypes.push_back(makeUnion(scope, location, leftDiscriminantTy, rightDiscriminantTy));
|
2023-06-09 13:20:36 +01:00
|
|
|
dest.get(def)->shouldAppendNilType |= partition.shouldAppendNilType || rhsIt->second.shouldAppendNilType;
|
2022-11-04 17:02:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-16 01:25:31 +00:00
|
|
|
void ConstraintGenerator::computeRefinement(const ScopePtr& scope, Location location, RefinementId refinement, RefinementContext* refis, bool sense,
|
|
|
|
bool eq, std::vector<ConstraintV>* constraints)
|
2022-11-04 17:02:37 +00:00
|
|
|
{
|
2023-02-03 12:34:12 +00:00
|
|
|
if (!refinement)
|
2022-11-04 17:02:37 +00:00
|
|
|
return;
|
2023-02-10 18:50:54 +00:00
|
|
|
else if (auto variadic = get<Variadic>(refinement))
|
|
|
|
{
|
|
|
|
for (RefinementId refi : variadic->refinements)
|
2024-01-27 02:30:40 +00:00
|
|
|
computeRefinement(scope, location, refi, refis, sense, eq, constraints);
|
2023-02-10 18:50:54 +00:00
|
|
|
}
|
2023-02-03 12:34:12 +00:00
|
|
|
else if (auto negation = get<Negation>(refinement))
|
2024-01-27 02:30:40 +00:00
|
|
|
return computeRefinement(scope, location, negation->refinement, refis, !sense, eq, constraints);
|
2023-02-03 12:34:12 +00:00
|
|
|
else if (auto conjunction = get<Conjunction>(refinement))
|
2022-11-04 17:02:37 +00:00
|
|
|
{
|
2023-03-03 13:45:38 +00:00
|
|
|
RefinementContext lhsRefis;
|
|
|
|
RefinementContext rhsRefis;
|
2022-11-04 17:02:37 +00:00
|
|
|
|
2024-01-27 02:30:40 +00:00
|
|
|
computeRefinement(scope, location, conjunction->lhs, sense ? refis : &lhsRefis, sense, eq, constraints);
|
|
|
|
computeRefinement(scope, location, conjunction->rhs, sense ? refis : &rhsRefis, sense, eq, constraints);
|
2022-11-04 17:02:37 +00:00
|
|
|
|
|
|
|
if (!sense)
|
2024-01-27 02:30:40 +00:00
|
|
|
unionRefinements(scope, location, lhsRefis, rhsRefis, *refis, constraints);
|
2022-11-04 17:02:37 +00:00
|
|
|
}
|
2023-02-03 12:34:12 +00:00
|
|
|
else if (auto disjunction = get<Disjunction>(refinement))
|
2022-11-04 17:02:37 +00:00
|
|
|
{
|
2023-03-03 13:45:38 +00:00
|
|
|
RefinementContext lhsRefis;
|
|
|
|
RefinementContext rhsRefis;
|
2022-11-04 17:02:37 +00:00
|
|
|
|
2024-01-27 02:30:40 +00:00
|
|
|
computeRefinement(scope, location, disjunction->lhs, sense ? &lhsRefis : refis, sense, eq, constraints);
|
|
|
|
computeRefinement(scope, location, disjunction->rhs, sense ? &rhsRefis : refis, sense, eq, constraints);
|
2022-11-04 17:02:37 +00:00
|
|
|
|
|
|
|
if (sense)
|
2024-01-27 02:30:40 +00:00
|
|
|
unionRefinements(scope, location, lhsRefis, rhsRefis, *refis, constraints);
|
2022-11-04 17:02:37 +00:00
|
|
|
}
|
2023-02-03 12:34:12 +00:00
|
|
|
else if (auto equivalence = get<Equivalence>(refinement))
|
2022-11-04 17:02:37 +00:00
|
|
|
{
|
2024-01-27 02:30:40 +00:00
|
|
|
computeRefinement(scope, location, equivalence->lhs, refis, sense, true, constraints);
|
|
|
|
computeRefinement(scope, location, equivalence->rhs, refis, sense, true, constraints);
|
2022-11-04 17:02:37 +00:00
|
|
|
}
|
2023-02-03 12:34:12 +00:00
|
|
|
else if (auto proposition = get<Proposition>(refinement))
|
2022-11-04 17:02:37 +00:00
|
|
|
{
|
|
|
|
TypeId discriminantTy = proposition->discriminantTy;
|
|
|
|
if (!sense && !eq)
|
2023-01-03 17:33:19 +00:00
|
|
|
discriminantTy = arena->addType(NegationType{proposition->discriminantTy});
|
2022-12-02 10:46:05 +00:00
|
|
|
else if (eq)
|
2022-11-04 17:02:37 +00:00
|
|
|
{
|
2023-01-03 17:33:19 +00:00
|
|
|
discriminantTy = arena->addType(BlockedType{});
|
2022-12-02 10:46:05 +00:00
|
|
|
constraints->push_back(SingletonOrTopTypeConstraint{discriminantTy, proposition->discriminantTy, !sense});
|
2022-11-04 17:02:37 +00:00
|
|
|
}
|
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
for (const RefinementKey* key = proposition->key; key; key = key->parent)
|
2023-03-03 13:45:38 +00:00
|
|
|
{
|
2023-10-20 21:36:26 +01:00
|
|
|
refis->insert(key->def, {});
|
|
|
|
refis->get(key->def)->discriminantTypes.push_back(discriminantTy);
|
2022-12-02 10:46:05 +00:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
// Reached leaf node
|
|
|
|
if (!key->propName)
|
|
|
|
break;
|
2023-06-09 13:20:36 +01:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
TypeId nextDiscriminantTy = arena->addType(TableType{});
|
|
|
|
NotNull<TableType> table{getMutable<TableType>(nextDiscriminantTy)};
|
2024-02-23 18:40:00 +00:00
|
|
|
table->props[*key->propName] = Property::readonly(discriminantTy);
|
2023-10-20 21:36:26 +01:00
|
|
|
table->scope = scope.get();
|
|
|
|
table->state = TableState::Sealed;
|
2022-12-02 10:46:05 +00:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
discriminantTy = nextDiscriminantTy;
|
2023-03-03 13:45:38 +00:00
|
|
|
}
|
2023-10-20 21:36:26 +01:00
|
|
|
|
|
|
|
// 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);
|
2023-03-03 13:45:38 +00:00
|
|
|
}
|
2022-12-02 10:46:05 +00:00
|
|
|
}
|
|
|
|
|
2023-05-19 19:59:59 +01:00
|
|
|
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
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
void ConstraintGenerator::applyRefinements(const ScopePtr& scope, Location location, RefinementId refinement)
|
2022-11-04 17:02:37 +00:00
|
|
|
{
|
2023-02-03 12:34:12 +00:00
|
|
|
if (!refinement)
|
2022-11-04 17:02:37 +00:00
|
|
|
return;
|
|
|
|
|
2023-03-03 13:45:38 +00:00
|
|
|
RefinementContext refinements;
|
2022-12-02 10:46:05 +00:00
|
|
|
std::vector<ConstraintV> constraints;
|
2024-01-27 02:30:40 +00:00
|
|
|
computeRefinement(scope, location, refinement, &refinements, /*sense*/ true, /*eq*/ false, &constraints);
|
2022-11-04 17:02:37 +00:00
|
|
|
|
2023-03-03 13:45:38 +00:00
|
|
|
for (auto& [def, partition] : refinements)
|
2022-11-04 17:02:37 +00:00
|
|
|
{
|
2024-01-27 02:30:40 +00:00
|
|
|
if (std::optional<TypeId> defTy = lookup(scope, def))
|
2023-03-03 13:45:38 +00:00
|
|
|
{
|
|
|
|
TypeId ty = *defTy;
|
|
|
|
if (partition.shouldAppendNilType)
|
|
|
|
ty = arena->addType(UnionType{{ty, builtinTypes->nilType}});
|
2022-11-04 17:02:37 +00:00
|
|
|
|
2023-05-19 19:59:59 +01:00
|
|
|
// 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))
|
|
|
|
{
|
2023-12-15 20:52:08 +00:00
|
|
|
TypeId resultType = arena->addType(TypeFamilyInstanceType{
|
|
|
|
NotNull{&kBuiltinTypeFamilies.refineFamily},
|
|
|
|
{ty, dt},
|
|
|
|
{},
|
|
|
|
});
|
|
|
|
addConstraint(scope, location, ReduceConstraint{resultType});
|
|
|
|
|
|
|
|
ty = resultType;
|
2023-05-19 19:59:59 +01:00
|
|
|
}
|
|
|
|
else
|
2023-08-04 18:01:35 +01:00
|
|
|
{
|
|
|
|
switch (shouldSuppressErrors(normalizer, ty))
|
|
|
|
{
|
2023-09-15 17:27:45 +01:00
|
|
|
case ErrorSuppression::DoNotSuppress:
|
2024-01-27 02:30:40 +00:00
|
|
|
ty = makeIntersect(scope, location, ty, dt);
|
2023-09-15 17:27:45 +01:00
|
|
|
break;
|
|
|
|
case ErrorSuppression::Suppress:
|
2024-01-27 02:30:40 +00:00
|
|
|
ty = makeIntersect(scope, location, ty, dt);
|
|
|
|
ty = makeUnion(scope, location, ty, builtinTypes->errorType);
|
2023-09-15 17:27:45 +01:00
|
|
|
break;
|
|
|
|
case ErrorSuppression::NormalizationFailed:
|
|
|
|
reportError(location, NormalizationTooComplex{});
|
2024-01-27 02:30:40 +00:00
|
|
|
ty = makeIntersect(scope, location, ty, dt);
|
2023-09-15 17:27:45 +01:00
|
|
|
break;
|
2023-08-04 18:01:35 +01:00
|
|
|
}
|
|
|
|
}
|
2023-05-19 19:59:59 +01:00
|
|
|
}
|
|
|
|
|
2023-10-06 18:31:16 +01:00
|
|
|
scope->rvalueRefinements[def] = ty;
|
2023-03-03 13:45:38 +00:00
|
|
|
}
|
2022-11-04 17:02:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (auto& c : constraints)
|
|
|
|
addConstraint(scope, location, c);
|
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ControlFlow ConstraintGenerator::visitBlockWithoutChildScope(const ScopePtr& scope, AstStatBlock* block)
|
2022-07-01 00:29:02 +01:00
|
|
|
{
|
|
|
|
RecursionCounter counter{&recursionCount};
|
|
|
|
|
|
|
|
if (recursionCount >= FInt::LuauCheckRecursionLimit)
|
|
|
|
{
|
|
|
|
reportCodeTooComplex(block->location);
|
2023-03-17 14:59:30 +00:00
|
|
|
return ControlFlow::None;
|
2022-07-01 00:29:02 +01:00
|
|
|
}
|
|
|
|
|
2022-08-04 22:27:28 +01:00
|
|
|
std::unordered_map<Name, Location> aliasDefinitionLocations;
|
|
|
|
|
|
|
|
// In order to enable mutually-recursive type aliases, we need to
|
|
|
|
// populate the type bindings before we actually check any of the
|
2023-02-17 14:53:37 +00:00
|
|
|
// alias statements.
|
2022-08-04 22:27:28 +01:00
|
|
|
for (AstStat* stat : block->body)
|
|
|
|
{
|
|
|
|
if (auto alias = stat->as<AstStatTypeAlias>())
|
|
|
|
{
|
2023-02-17 14:53:37 +00:00
|
|
|
if (scope->exportedTypeBindings.count(alias->name.value) || scope->privateTypeBindings.count(alias->name.value))
|
2022-08-04 22:27:28 +01:00
|
|
|
{
|
|
|
|
auto it = aliasDefinitionLocations.find(alias->name.value);
|
|
|
|
LUAU_ASSERT(it != aliasDefinitionLocations.end());
|
|
|
|
reportError(alias->location, DuplicateTypeDefinition{alias->name.value, it->second});
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2024-01-12 19:16:39 +00:00
|
|
|
// 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;
|
|
|
|
|
2023-02-17 14:53:37 +00:00
|
|
|
ScopePtr defnScope = childScope(alias, scope);
|
2022-08-04 22:27:28 +01:00
|
|
|
|
2023-02-17 14:53:37 +00:00
|
|
|
TypeId initialType = arena->addType(BlockedType{});
|
|
|
|
TypeFun initialFun{initialType};
|
2022-08-04 22:27:28 +01:00
|
|
|
|
2023-02-17 14:53:37 +00:00
|
|
|
for (const auto& [name, gen] : createGenerics(defnScope, alias->generics, /* useCache */ true))
|
2022-08-04 22:27:28 +01:00
|
|
|
{
|
|
|
|
initialFun.typeParams.push_back(gen);
|
|
|
|
}
|
|
|
|
|
2023-02-17 14:53:37 +00:00
|
|
|
for (const auto& [name, genPack] : createGenericPacks(defnScope, alias->genericPacks, /* useCache */ true))
|
2022-08-04 22:27:28 +01:00
|
|
|
{
|
|
|
|
initialFun.typePackParams.push_back(genPack);
|
|
|
|
}
|
|
|
|
|
2023-02-17 14:53:37 +00:00
|
|
|
if (alias->exported)
|
|
|
|
scope->exportedTypeBindings[alias->name.value] = std::move(initialFun);
|
|
|
|
else
|
|
|
|
scope->privateTypeBindings[alias->name.value] = std::move(initialFun);
|
|
|
|
|
2022-08-04 22:27:28 +01:00
|
|
|
astTypeAliasDefiningScopes[alias] = defnScope;
|
|
|
|
aliasDefinitionLocations[alias->name.value] = alias->location;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
std::optional<ControlFlow> firstControlFlow;
|
2022-07-01 00:29:02 +01:00
|
|
|
for (AstStat* stat : block->body)
|
2023-03-17 14:59:30 +00:00
|
|
|
{
|
|
|
|
ControlFlow cf = visit(scope, stat);
|
|
|
|
if (cf != ControlFlow::None && !firstControlFlow)
|
|
|
|
firstControlFlow = cf;
|
|
|
|
}
|
|
|
|
|
|
|
|
return firstControlFlow.value_or(ControlFlow::None);
|
2022-06-03 21:32:20 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStat* stat)
|
2022-06-03 21:32:20 +01:00
|
|
|
{
|
2022-07-01 00:29:02 +01:00
|
|
|
RecursionLimiter limiter{&recursionCount, FInt::LuauCheckRecursionLimit};
|
2022-06-03 21:32:20 +01:00
|
|
|
|
|
|
|
if (auto s = stat->as<AstStatBlock>())
|
2023-03-17 14:59:30 +00:00
|
|
|
return visit(scope, s);
|
2023-02-17 14:53:37 +00:00
|
|
|
else if (auto i = stat->as<AstStatIf>())
|
2023-03-17 14:59:30 +00:00
|
|
|
return visit(scope, i);
|
2022-08-11 21:42:54 +01:00
|
|
|
else if (auto s = stat->as<AstStatWhile>())
|
2023-03-17 14:59:30 +00:00
|
|
|
return visit(scope, s);
|
2022-08-18 22:04:33 +01:00
|
|
|
else if (auto s = stat->as<AstStatRepeat>())
|
2023-03-17 14:59:30 +00:00
|
|
|
return visit(scope, s);
|
2023-09-30 01:22:06 +01:00
|
|
|
else if (stat->is<AstStatBreak>())
|
|
|
|
return FFlag::LuauLoopControlFlowAnalysis ? ControlFlow::Breaks : ControlFlow::None;
|
|
|
|
else if (stat->is<AstStatContinue>())
|
|
|
|
return FFlag::LuauLoopControlFlowAnalysis ? ControlFlow::Continues : ControlFlow::None;
|
2022-06-03 21:32:20 +01:00
|
|
|
else if (auto r = stat->as<AstStatReturn>())
|
2023-03-17 14:59:30 +00:00
|
|
|
return visit(scope, r);
|
2023-02-17 14:53:37 +00:00
|
|
|
else if (auto e = stat->as<AstStatExpr>())
|
2023-03-17 14:59:30 +00:00
|
|
|
{
|
2023-02-17 14:53:37 +00:00
|
|
|
checkPack(scope, e->expr);
|
2023-03-17 14:59:30 +00:00
|
|
|
|
|
|
|
if (auto call = e->expr->as<AstExprCall>(); call && doesCallError(call))
|
|
|
|
return ControlFlow::Throws;
|
|
|
|
|
|
|
|
return ControlFlow::None;
|
|
|
|
}
|
2023-02-17 14:53:37 +00:00
|
|
|
else if (auto s = stat->as<AstStatLocal>())
|
2023-03-17 14:59:30 +00:00
|
|
|
return visit(scope, s);
|
2023-02-17 14:53:37 +00:00
|
|
|
else if (auto s = stat->as<AstStatFor>())
|
2023-03-17 14:59:30 +00:00
|
|
|
return visit(scope, s);
|
2023-02-17 14:53:37 +00:00
|
|
|
else if (auto s = stat->as<AstStatForIn>())
|
2023-03-17 14:59:30 +00:00
|
|
|
return visit(scope, s);
|
2022-06-17 01:54:42 +01:00
|
|
|
else if (auto a = stat->as<AstStatAssign>())
|
2023-03-17 14:59:30 +00:00
|
|
|
return visit(scope, a);
|
2022-08-18 22:04:33 +01:00
|
|
|
else if (auto a = stat->as<AstStatCompoundAssign>())
|
2023-03-17 14:59:30 +00:00
|
|
|
return visit(scope, a);
|
2023-02-17 14:53:37 +00:00
|
|
|
else if (auto f = stat->as<AstStatFunction>())
|
2023-03-17 14:59:30 +00:00
|
|
|
return visit(scope, f);
|
2023-02-17 14:53:37 +00:00
|
|
|
else if (auto f = stat->as<AstStatLocalFunction>())
|
2023-03-17 14:59:30 +00:00
|
|
|
return visit(scope, f);
|
2022-06-24 02:44:07 +01:00
|
|
|
else if (auto a = stat->as<AstStatTypeAlias>())
|
2023-03-17 14:59:30 +00:00
|
|
|
return visit(scope, a);
|
2022-08-04 22:27:28 +01:00
|
|
|
else if (auto s = stat->as<AstStatDeclareGlobal>())
|
2023-03-17 14:59:30 +00:00
|
|
|
return visit(scope, s);
|
2022-08-04 22:27:28 +01:00
|
|
|
else if (auto s = stat->as<AstStatDeclareFunction>())
|
2023-03-17 14:59:30 +00:00
|
|
|
return visit(scope, s);
|
2023-02-17 14:53:37 +00:00
|
|
|
else if (auto s = stat->as<AstStatDeclareClass>())
|
2023-03-17 14:59:30 +00:00
|
|
|
return visit(scope, s);
|
2022-09-29 23:11:54 +01:00
|
|
|
else if (auto s = stat->as<AstStatError>())
|
2023-03-17 14:59:30 +00:00
|
|
|
return visit(scope, s);
|
2022-06-03 21:32:20 +01:00
|
|
|
else
|
2023-03-17 14:59:30 +00:00
|
|
|
{
|
2023-02-17 14:53:37 +00:00
|
|
|
LUAU_ASSERT(0 && "Internal error: Unknown AstStat type");
|
2023-03-17 14:59:30 +00:00
|
|
|
return ControlFlow::None;
|
|
|
|
}
|
2022-06-03 21:32:20 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatLocal* statLocal)
|
2022-06-03 21:32:20 +01:00
|
|
|
{
|
2023-11-17 18:15:31 +00:00
|
|
|
std::vector<TypeId> annotatedTypes;
|
|
|
|
annotatedTypes.reserve(statLocal->vars.size);
|
|
|
|
bool hasAnnotation = false;
|
|
|
|
|
|
|
|
std::vector<std::optional<TypeId>> expectedTypes;
|
|
|
|
expectedTypes.reserve(statLocal->vars.size);
|
2022-06-03 21:32:20 +01:00
|
|
|
|
2023-10-13 20:38:31 +01:00
|
|
|
std::vector<TypeId> assignees;
|
2023-10-20 21:36:26 +01:00
|
|
|
assignees.reserve(statLocal->vars.size);
|
2023-10-13 20:38:31 +01:00
|
|
|
|
2023-01-27 21:28:45 +00:00
|
|
|
// Used to name the first value type, even if it's not placed in varTypes,
|
|
|
|
// for the purpose of synthetic name attribution.
|
|
|
|
std::optional<TypeId> firstValueType;
|
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
for (AstLocal* local : statLocal->vars)
|
2022-06-03 21:32:20 +01:00
|
|
|
{
|
2023-10-20 21:36:26 +01:00
|
|
|
const Location location = local->location;
|
|
|
|
|
2023-11-17 18:15:31 +00:00
|
|
|
TypeId assignee = arena->addType(LocalType{builtinTypes->neverType, /* blockCount */ 1, local->name.value});
|
|
|
|
|
2023-10-13 20:38:31 +01:00
|
|
|
assignees.push_back(assignee);
|
2022-09-23 19:32:10 +01:00
|
|
|
|
2023-10-13 20:38:31 +01:00
|
|
|
if (!firstValueType)
|
|
|
|
firstValueType = assignee;
|
2022-10-21 18:33:43 +01:00
|
|
|
|
2023-10-13 20:38:31 +01:00
|
|
|
if (local->annotation)
|
2022-11-04 17:02:37 +00:00
|
|
|
{
|
2023-11-17 18:15:31 +00:00
|
|
|
hasAnnotation = true;
|
2023-10-13 20:38:31 +01:00
|
|
|
TypeId annotationTy = resolveType(scope, local->annotation, /* inTypeArguments */ false);
|
2023-11-17 18:15:31 +00:00
|
|
|
annotatedTypes.push_back(annotationTy);
|
|
|
|
expectedTypes.push_back(annotationTy);
|
2023-10-20 21:36:26 +01:00
|
|
|
|
|
|
|
scope->bindings[local] = Binding{annotationTy, location};
|
2022-11-04 17:02:37 +00:00
|
|
|
}
|
|
|
|
else
|
2023-10-20 21:36:26 +01:00
|
|
|
{
|
2023-11-17 18:15:31 +00:00
|
|
|
// 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};
|
2022-06-17 01:54:42 +01:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
inferredBindings[local] = {scope.get(), location, {assignee}};
|
|
|
|
}
|
|
|
|
|
|
|
|
DefId def = dfg->getDef(local);
|
|
|
|
scope->lvalueTypes[def] = assignee;
|
2022-09-02 00:00:14 +01:00
|
|
|
}
|
|
|
|
|
2023-11-17 18:15:31 +00:00
|
|
|
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))});
|
2023-10-13 20:38:31 +01:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
if (statLocal->vars.size == 1 && statLocal->values.size == 1 && firstValueType && scope.get() == rootScope)
|
2023-01-27 21:28:45 +00:00
|
|
|
{
|
2023-10-20 21:36:26 +01:00
|
|
|
AstLocal* var = statLocal->vars.data[0];
|
|
|
|
AstExpr* value = statLocal->values.data[0];
|
2023-01-27 21:28:45 +00:00
|
|
|
|
|
|
|
if (value->is<AstExprTable>())
|
|
|
|
addConstraint(scope, value->location, NameConstraint{*firstValueType, var->name.value, /*synthetic*/ true});
|
|
|
|
else if (const AstExprCall* call = value->as<AstExprCall>())
|
|
|
|
{
|
|
|
|
if (const AstExprGlobal* global = call->func->as<AstExprGlobal>(); global && global->name == "setmetatable")
|
|
|
|
{
|
|
|
|
addConstraint(scope, value->location, NameConstraint{*firstValueType, var->name.value, /*synthetic*/ true});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
if (statLocal->values.size > 0)
|
2022-09-02 00:00:14 +01:00
|
|
|
{
|
|
|
|
// To correctly handle 'require', we need to import the exported type bindings into the variable 'namespace'.
|
2023-10-20 21:36:26 +01:00
|
|
|
for (size_t i = 0; i < statLocal->values.size && i < statLocal->vars.size; ++i)
|
2022-09-02 00:00:14 +01:00
|
|
|
{
|
2023-10-20 21:36:26 +01:00
|
|
|
const AstExprCall* call = statLocal->values.data[i]->as<AstExprCall>();
|
2022-09-02 00:00:14 +01:00
|
|
|
if (!call)
|
|
|
|
continue;
|
|
|
|
|
2023-10-13 20:38:31 +01:00
|
|
|
auto maybeRequire = matchRequire(*call);
|
|
|
|
if (!maybeRequire)
|
|
|
|
continue;
|
2022-09-02 00:00:14 +01:00
|
|
|
|
2023-10-13 20:38:31 +01:00
|
|
|
AstExpr* require = *maybeRequire;
|
2022-09-02 00:00:14 +01:00
|
|
|
|
2023-10-13 20:38:31 +01:00
|
|
|
auto moduleInfo = moduleResolver->resolveModuleInfo(module->name, *require);
|
|
|
|
if (!moduleInfo)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
ModulePtr module = moduleResolver->getModule(moduleInfo->name);
|
|
|
|
if (!module)
|
|
|
|
continue;
|
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
const Name name{statLocal->vars.data[i]->name.value};
|
2023-10-13 20:38:31 +01:00
|
|
|
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};
|
2022-09-02 00:00:14 +01:00
|
|
|
}
|
2022-06-03 21:32:20 +01:00
|
|
|
}
|
|
|
|
}
|
2023-03-17 14:59:30 +00:00
|
|
|
|
|
|
|
return ControlFlow::None;
|
2022-06-03 21:32:20 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatFor* for_)
|
2022-07-14 23:39:35 +01:00
|
|
|
{
|
2023-03-03 13:45:38 +00:00
|
|
|
TypeId annotationTy = builtinTypes->numberType;
|
2023-02-03 12:34:12 +00:00
|
|
|
if (for_->var->annotation)
|
2023-03-03 13:45:38 +00:00
|
|
|
annotationTy = resolveType(scope, for_->var->annotation, /* inTypeArguments */ false);
|
2023-02-03 12:34:12 +00:00
|
|
|
|
|
|
|
auto inferNumber = [&](AstExpr* expr) {
|
2022-07-14 23:39:35 +01:00
|
|
|
if (!expr)
|
|
|
|
return;
|
2022-08-04 22:27:28 +01:00
|
|
|
|
2022-10-27 23:22:49 +01:00
|
|
|
TypeId t = check(scope, expr).ty;
|
2023-01-03 17:33:19 +00:00
|
|
|
addConstraint(scope, expr->location, SubtypeConstraint{t, builtinTypes->numberType});
|
2022-07-14 23:39:35 +01:00
|
|
|
};
|
|
|
|
|
2023-02-03 12:34:12 +00:00
|
|
|
inferNumber(for_->from);
|
|
|
|
inferNumber(for_->to);
|
|
|
|
inferNumber(for_->step);
|
2022-07-14 23:39:35 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
ScopePtr forScope = childScope(for_, scope);
|
2023-03-03 13:45:38 +00:00
|
|
|
forScope->bindings[for_->var] = Binding{annotationTy, for_->var->location};
|
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
DefId def = dfg->getDef(for_->var);
|
|
|
|
forScope->lvalueTypes[def] = annotationTy;
|
|
|
|
forScope->rvalueRefinements[def] = annotationTy;
|
2022-07-14 23:39:35 +01:00
|
|
|
|
|
|
|
visit(forScope, for_->body);
|
2023-03-17 14:59:30 +00:00
|
|
|
|
|
|
|
return ControlFlow::None;
|
2022-07-14 23:39:35 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatForIn* forIn)
|
2022-09-02 00:00:14 +01:00
|
|
|
{
|
|
|
|
ScopePtr loopScope = childScope(forIn, scope);
|
|
|
|
|
2022-10-27 23:22:49 +01:00
|
|
|
TypePackId iterator = checkPack(scope, forIn->values).tp;
|
2022-09-02 00:00:14 +01:00
|
|
|
|
|
|
|
std::vector<TypeId> variableTypes;
|
|
|
|
variableTypes.reserve(forIn->vars.size);
|
2023-10-13 20:38:31 +01:00
|
|
|
|
2022-09-02 00:00:14 +01:00
|
|
|
for (AstLocal* var : forIn->vars)
|
|
|
|
{
|
2024-01-27 02:30:40 +00:00
|
|
|
TypeId assignee = arena->addType(LocalType{builtinTypes->neverType, /* blockCount */ 1, var->name.value});
|
2023-10-13 20:38:31 +01:00
|
|
|
variableTypes.push_back(assignee);
|
2022-11-10 22:04:44 +00:00
|
|
|
|
2023-12-02 02:04:44 +00:00
|
|
|
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};
|
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
DefId def = dfg->getDef(var);
|
|
|
|
loopScope->lvalueTypes[def] = assignee;
|
2022-09-02 00:00:14 +01:00
|
|
|
}
|
|
|
|
|
2023-10-13 20:38:31 +01:00
|
|
|
TypePackId variablePack = arena->addTypePack(std::move(variableTypes));
|
2023-05-25 21:46:51 +01:00
|
|
|
addConstraint(
|
2023-06-24 06:33:44 +01:00
|
|
|
loopScope, getLocation(forIn->values), IterableConstraint{iterator, variablePack, forIn->values.data[0], &module->astForInNextTypes});
|
2023-10-13 20:38:31 +01:00
|
|
|
|
2022-09-02 00:00:14 +01:00
|
|
|
visit(loopScope, forIn->body);
|
2023-03-17 14:59:30 +00:00
|
|
|
|
|
|
|
return ControlFlow::None;
|
2022-09-02 00:00:14 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatWhile* while_)
|
2022-08-11 21:42:54 +01:00
|
|
|
{
|
2023-07-07 18:14:35 +01:00
|
|
|
RefinementId refinement = check(scope, while_->condition).refinement;
|
2022-08-11 21:42:54 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
ScopePtr whileScope = childScope(while_, scope);
|
2023-07-07 18:14:35 +01:00
|
|
|
applyRefinements(whileScope, while_->condition->location, refinement);
|
2022-08-11 21:42:54 +01:00
|
|
|
|
|
|
|
visit(whileScope, while_->body);
|
2023-03-17 14:59:30 +00:00
|
|
|
|
|
|
|
return ControlFlow::None;
|
2022-08-11 21:42:54 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatRepeat* repeat)
|
2022-08-18 22:04:33 +01:00
|
|
|
{
|
|
|
|
ScopePtr repeatScope = childScope(repeat, scope);
|
|
|
|
|
2023-03-03 13:45:38 +00:00
|
|
|
visitBlockWithoutChildScope(repeatScope, repeat->body);
|
2022-08-18 22:04:33 +01:00
|
|
|
|
|
|
|
check(repeatScope, repeat->condition);
|
2023-03-17 14:59:30 +00:00
|
|
|
|
|
|
|
return ControlFlow::None;
|
2022-08-18 22:04:33 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatLocalFunction* function)
|
2022-06-03 21:32:20 +01:00
|
|
|
{
|
|
|
|
// Local
|
|
|
|
// Global
|
|
|
|
// Dotted path
|
|
|
|
// Self?
|
|
|
|
|
|
|
|
TypeId functionType = nullptr;
|
|
|
|
auto ty = scope->lookup(function->name);
|
2022-07-01 00:29:02 +01:00
|
|
|
LUAU_ASSERT(!ty.has_value()); // The parser ensures that every local function has a distinct Symbol for its name.
|
|
|
|
|
2023-01-03 17:33:19 +00:00
|
|
|
functionType = arena->addType(BlockedType{});
|
2022-07-29 04:41:13 +01:00
|
|
|
scope->bindings[function->name] = Binding{functionType, function->name->location};
|
2022-06-03 21:32:20 +01:00
|
|
|
|
2023-05-19 19:59:59 +01:00
|
|
|
FunctionSignature sig = checkFunctionSignature(scope, function->func, /* expectedType */ std::nullopt, function->name->location);
|
2022-07-29 04:41:13 +01:00
|
|
|
sig.bodyScope->bindings[function->name] = Binding{sig.signature, function->func->location};
|
2022-06-03 21:32:20 +01:00
|
|
|
|
2024-02-09 17:32:52 +00:00
|
|
|
bool sigFullyDefined = !hasFreeType(sig.signature);
|
|
|
|
if (sigFullyDefined)
|
|
|
|
asMutable(functionType)->ty.emplace<BoundType>(sig.signature);
|
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
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;
|
2023-03-03 13:45:38 +00:00
|
|
|
|
2022-12-02 10:46:05 +00:00
|
|
|
Checkpoint start = checkpoint(this);
|
2022-07-01 00:29:02 +01:00
|
|
|
checkFunctionBody(sig.bodyScope, function->func);
|
2022-12-02 10:46:05 +00:00
|
|
|
Checkpoint end = checkpoint(this);
|
2022-06-03 21:32:20 +01:00
|
|
|
|
2024-02-09 17:32:52 +00:00
|
|
|
if (!sigFullyDefined)
|
|
|
|
{
|
|
|
|
NotNull<Scope> constraintScope{sig.signatureScope ? sig.signatureScope.get() : sig.bodyScope.get()};
|
|
|
|
std::unique_ptr<Constraint> c =
|
|
|
|
std::make_unique<Constraint>(constraintScope, function->name->location, GeneralizationConstraint{functionType, sig.signature});
|
2022-11-10 22:04:44 +00:00
|
|
|
|
2024-02-09 17:32:52 +00:00
|
|
|
Constraint* previous = nullptr;
|
2024-02-16 01:25:31 +00:00
|
|
|
forEachConstraint(start, end, this, [&c, &previous](const ConstraintPtr& constraint) {
|
|
|
|
c->dependencies.push_back(NotNull{constraint.get()});
|
2023-07-07 18:14:35 +01:00
|
|
|
|
2024-02-16 01:25:31 +00:00
|
|
|
if (auto psc = get<PackSubtypeConstraint>(*constraint); psc && psc->returns)
|
|
|
|
{
|
|
|
|
if (previous)
|
|
|
|
constraint->dependencies.push_back(NotNull{previous});
|
2023-07-07 18:14:35 +01:00
|
|
|
|
2024-02-16 01:25:31 +00:00
|
|
|
previous = constraint.get();
|
|
|
|
}
|
|
|
|
});
|
2022-06-17 01:54:42 +01:00
|
|
|
|
2024-02-09 17:32:52 +00:00
|
|
|
addConstraint(scope, std::move(c));
|
|
|
|
module->astTypes[function->func] = functionType;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
module->astTypes[function->func] = sig.signature;
|
2023-03-17 14:59:30 +00:00
|
|
|
|
|
|
|
return ControlFlow::None;
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatFunction* function)
|
2022-06-17 01:54:42 +01:00
|
|
|
{
|
|
|
|
// Name could be AstStatLocal, AstStatGlobal, AstStatIndexName.
|
|
|
|
// With or without self
|
|
|
|
|
2023-01-03 17:33:19 +00:00
|
|
|
TypeId generalizedType = arena->addType(BlockedType{});
|
2022-06-17 01:54:42 +01:00
|
|
|
|
2023-02-17 14:53:37 +00:00
|
|
|
Checkpoint start = checkpoint(this);
|
2023-05-19 19:59:59 +01:00
|
|
|
FunctionSignature sig = checkFunctionSignature(scope, function->func, /* expectedType */ std::nullopt, function->name->location);
|
2024-02-09 17:32:52 +00:00
|
|
|
bool sigFullyDefined = !hasFreeType(sig.signature);
|
|
|
|
|
|
|
|
if (sigFullyDefined)
|
|
|
|
asMutable(generalizedType)->ty.emplace<BoundType>(sig.signature);
|
2022-06-03 21:32:20 +01:00
|
|
|
|
2023-11-10 18:05:48 +00:00
|
|
|
DenseHashSet<Constraint*> excludeList{nullptr};
|
2023-02-24 18:24:22 +00:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
DefId def = dfg->getDef(function->name);
|
2024-01-27 02:30:40 +00:00
|
|
|
std::optional<TypeId> existingFunctionTy = lookup(scope, def);
|
2023-10-06 18:31:16 +01:00
|
|
|
|
2024-02-09 17:32:52 +00:00
|
|
|
if (sigFullyDefined && existingFunctionTy && get<BlockedType>(*existingFunctionTy))
|
|
|
|
asMutable(*existingFunctionTy)->ty.emplace<BoundType>(sig.signature);
|
|
|
|
|
2022-06-17 01:54:42 +01:00
|
|
|
if (AstExprLocal* localName = function->name->as<AstExprLocal>())
|
2022-06-03 21:32:20 +01:00
|
|
|
{
|
2022-06-17 01:54:42 +01:00
|
|
|
if (existingFunctionTy)
|
|
|
|
{
|
2022-12-02 10:46:05 +00:00
|
|
|
addConstraint(scope, function->name->location, SubtypeConstraint{generalizedType, *existingFunctionTy});
|
|
|
|
|
|
|
|
Symbol sym{localName->local};
|
|
|
|
scope->bindings[sym].typeId = generalizedType;
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
else
|
2022-12-02 10:46:05 +00:00
|
|
|
scope->bindings[localName->local] = Binding{generalizedType, localName->location};
|
|
|
|
|
2022-07-29 04:41:13 +01:00
|
|
|
sig.bodyScope->bindings[localName->local] = Binding{sig.signature, localName->location};
|
2023-10-20 21:36:26 +01:00
|
|
|
sig.bodyScope->lvalueTypes[def] = sig.signature;
|
|
|
|
sig.bodyScope->rvalueRefinements[def] = sig.signature;
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
else if (AstExprGlobal* globalName = function->name->as<AstExprGlobal>())
|
|
|
|
{
|
2022-12-02 10:46:05 +00:00
|
|
|
if (!existingFunctionTy)
|
|
|
|
ice->ice("prepopulateGlobalScope did not populate a global name", globalName->location);
|
|
|
|
|
2024-02-09 17:32:52 +00:00
|
|
|
if (!sigFullyDefined)
|
|
|
|
generalizedType = *existingFunctionTy;
|
2022-12-02 10:46:05 +00:00
|
|
|
|
2022-07-29 04:41:13 +01:00
|
|
|
sig.bodyScope->bindings[globalName->name] = Binding{sig.signature, globalName->location};
|
2023-10-20 21:36:26 +01:00
|
|
|
sig.bodyScope->lvalueTypes[def] = sig.signature;
|
|
|
|
sig.bodyScope->rvalueRefinements[def] = sig.signature;
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
else if (AstExprIndexName* indexName = function->name->as<AstExprIndexName>())
|
|
|
|
{
|
2023-02-24 18:24:22 +00:00
|
|
|
Checkpoint check1 = checkpoint(this);
|
2023-11-03 19:47:28 +00:00
|
|
|
std::optional<TypeId> lvalueType = checkLValue(scope, indexName, generalizedType);
|
2023-10-20 21:36:26 +01:00
|
|
|
LUAU_ASSERT(lvalueType);
|
2023-02-24 18:24:22 +00:00
|
|
|
Checkpoint check2 = checkpoint(this);
|
|
|
|
|
|
|
|
forEachConstraint(check1, check2, this, [&excludeList](const ConstraintPtr& c) {
|
|
|
|
excludeList.insert(c.get());
|
|
|
|
});
|
|
|
|
|
2023-02-10 18:50:54 +00:00
|
|
|
// TODO figure out how to populate the location field of the table Property.
|
2023-02-24 18:24:22 +00:00
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
if (lvalueType && *lvalueType != generalizedType)
|
2023-10-20 21:36:26 +01:00
|
|
|
{
|
2023-11-03 19:47:28 +00:00
|
|
|
addConstraint(scope, indexName->location, SubtypeConstraint{*lvalueType, generalizedType});
|
2023-10-20 21:36:26 +01:00
|
|
|
}
|
2022-07-01 00:29:02 +01:00
|
|
|
}
|
|
|
|
else if (AstExprError* err = function->name->as<AstExprError>())
|
|
|
|
{
|
2023-01-03 17:33:19 +00:00
|
|
|
generalizedType = builtinTypes->errorRecoveryType();
|
2022-06-03 21:32:20 +01:00
|
|
|
}
|
|
|
|
|
2022-12-02 10:46:05 +00:00
|
|
|
if (generalizedType == nullptr)
|
|
|
|
ice->ice("generalizedType == nullptr", function->location);
|
2022-06-03 21:32:20 +01:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
scope->rvalueRefinements[def] = generalizedType;
|
2023-03-03 13:45:38 +00:00
|
|
|
|
2022-07-01 00:29:02 +01:00
|
|
|
checkFunctionBody(sig.bodyScope, function->func);
|
2022-12-02 10:46:05 +00:00
|
|
|
Checkpoint end = checkpoint(this);
|
2022-07-01 00:29:02 +01:00
|
|
|
|
2024-02-09 17:32:52 +00:00
|
|
|
if (!sigFullyDefined)
|
|
|
|
{
|
|
|
|
NotNull<Scope> constraintScope{sig.signatureScope ? sig.signatureScope.get() : sig.bodyScope.get()};
|
|
|
|
std::unique_ptr<Constraint> c =
|
|
|
|
std::make_unique<Constraint>(constraintScope, function->name->location, GeneralizationConstraint{generalizedType, sig.signature});
|
2022-11-10 22:04:44 +00:00
|
|
|
|
2024-02-09 17:32:52 +00:00
|
|
|
Constraint* previous = nullptr;
|
2024-02-16 01:25:31 +00:00
|
|
|
forEachConstraint(start, end, this, [&c, &excludeList, &previous](const ConstraintPtr& constraint) {
|
|
|
|
if (!excludeList.contains(constraint.get()))
|
|
|
|
c->dependencies.push_back(NotNull{constraint.get()});
|
2023-07-07 18:14:35 +01:00
|
|
|
|
2024-02-16 01:25:31 +00:00
|
|
|
if (auto psc = get<PackSubtypeConstraint>(*constraint); psc && psc->returns)
|
|
|
|
{
|
|
|
|
if (previous)
|
|
|
|
constraint->dependencies.push_back(NotNull{previous});
|
2023-07-07 18:14:35 +01:00
|
|
|
|
2024-02-16 01:25:31 +00:00
|
|
|
previous = constraint.get();
|
|
|
|
}
|
|
|
|
});
|
2022-06-03 21:32:20 +01:00
|
|
|
|
2024-02-09 17:32:52 +00:00
|
|
|
addConstraint(scope, std::move(c));
|
|
|
|
}
|
2023-03-17 14:59:30 +00:00
|
|
|
|
|
|
|
return ControlFlow::None;
|
2022-06-03 21:32:20 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatReturn* ret)
|
2022-06-03 21:32:20 +01:00
|
|
|
{
|
2022-09-23 19:32:10 +01:00
|
|
|
// 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.
|
2023-02-17 14:53:37 +00:00
|
|
|
std::vector<std::optional<TypeId>> expectedTypes;
|
2022-09-23 19:32:10 +01:00
|
|
|
for (TypeId ty : scope->returnType)
|
|
|
|
expectedTypes.push_back(ty);
|
|
|
|
|
2022-10-27 23:22:49 +01:00
|
|
|
TypePackId exprTypes = checkPack(scope, ret->list, expectedTypes).tp;
|
2023-07-07 18:14:35 +01:00
|
|
|
addConstraint(scope, ret->location, PackSubtypeConstraint{exprTypes, scope->returnType, /*returns*/ true});
|
2023-03-17 14:59:30 +00:00
|
|
|
|
|
|
|
return ControlFlow::Returns;
|
2022-06-03 21:32:20 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatBlock* block)
|
2022-06-03 21:32:20 +01:00
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
ScopePtr innerScope = childScope(block, scope);
|
2022-06-03 21:32:20 +01:00
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
ControlFlow flow = visitBlockWithoutChildScope(innerScope, block);
|
2023-11-10 18:05:48 +00:00
|
|
|
|
|
|
|
// 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.
|
2023-03-17 14:59:30 +00:00
|
|
|
scope->inheritRefinements(innerScope);
|
2023-11-10 18:05:48 +00:00
|
|
|
scope->inheritAssignments(innerScope);
|
2023-03-17 14:59:30 +00:00
|
|
|
|
|
|
|
return flow;
|
2022-06-03 21:32:20 +01:00
|
|
|
}
|
|
|
|
|
2023-08-04 18:01:35 +01:00
|
|
|
// TODO Clip?
|
2023-02-24 18:24:22 +00:00
|
|
|
static void bindFreeType(TypeId a, TypeId b)
|
2022-06-17 01:54:42 +01:00
|
|
|
{
|
2023-02-24 18:24:22 +00:00
|
|
|
FreeType* af = getMutable<FreeType>(a);
|
|
|
|
FreeType* bf = getMutable<FreeType>(b);
|
2022-12-02 10:46:05 +00:00
|
|
|
|
2023-02-24 18:24:22 +00:00
|
|
|
LUAU_ASSERT(af || bf);
|
|
|
|
|
|
|
|
if (!bf)
|
|
|
|
asMutable(a)->ty.emplace<BoundType>(b);
|
|
|
|
else if (!af)
|
|
|
|
asMutable(b)->ty.emplace<BoundType>(a);
|
|
|
|
else if (subsumes(bf->scope, af->scope))
|
|
|
|
asMutable(a)->ty.emplace<BoundType>(b);
|
|
|
|
else if (subsumes(af->scope, bf->scope))
|
|
|
|
asMutable(b)->ty.emplace<BoundType>(a);
|
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatAssign* assign)
|
2023-02-24 18:24:22 +00:00
|
|
|
{
|
2023-10-13 20:38:31 +01:00
|
|
|
std::vector<TypeId> assignees;
|
|
|
|
assignees.reserve(assign->vars.size);
|
|
|
|
|
2024-01-19 15:13:08 +00:00
|
|
|
size_t i = 0;
|
2023-10-13 20:38:31 +01:00
|
|
|
for (AstExpr* lvalue : assign->vars)
|
2023-02-17 14:53:37 +00:00
|
|
|
{
|
2023-10-13 20:38:31 +01:00
|
|
|
TypeId assignee = arena->addType(BlockedType{});
|
2022-06-17 01:54:42 +01:00
|
|
|
|
2024-01-19 15:13:08 +00:00
|
|
|
// This is a really weird thing to do, but it's critically important for some kinds of
|
|
|
|
// assignments with the current type state behavior. Consider this code:
|
|
|
|
// local function f(l, r)
|
|
|
|
// local i = l
|
|
|
|
// for _ = l, r do
|
|
|
|
// i = i + 1
|
|
|
|
// end
|
|
|
|
// end
|
|
|
|
//
|
|
|
|
// With type states now, we will not create a new state for `i` within the loop. This means
|
|
|
|
// that, in the absence of the analysis below, we would infer a too-broad bound for i: the
|
|
|
|
// cyclic type t1 where t1 = add<t1 | number, number>. In order to stop this, we say that
|
|
|
|
// assignments to a definition with a self-referential binary expression do not transform
|
|
|
|
// the type of the definition. This will only apply for loops, where the definition is
|
|
|
|
// shared in more places; for non-loops, there will be a separate DefId for the lvalue in
|
|
|
|
// the assignment, so we will deem the expression to be transformative.
|
|
|
|
//
|
|
|
|
// Deeming the addition in the code sample above as non-transformative means that i is known
|
|
|
|
// to be exactly number further on, ensuring the type family reduces down to number, as is
|
|
|
|
// expected for this code snippet.
|
|
|
|
//
|
|
|
|
// There is a potential for spurious errors here if the expression is more complex than a
|
|
|
|
// simple binary expression, e.g. i = (i + 1) * 2. At the time of writing, this case hasn't
|
|
|
|
// materialized.
|
|
|
|
bool transform = true;
|
|
|
|
|
|
|
|
if (assign->values.size > i)
|
|
|
|
{
|
|
|
|
AstExpr* value = assign->values.data[i];
|
|
|
|
if (auto bexp = value->as<AstExprBinary>())
|
|
|
|
{
|
|
|
|
DefId lvalueDef = dfg->getDef(lvalue);
|
|
|
|
DefId lDef = dfg->getDef(bexp->left);
|
|
|
|
DefId rDef = dfg->getDef(bexp->right);
|
|
|
|
|
|
|
|
if (lvalueDef == lDef || lvalueDef == rDef)
|
|
|
|
transform = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
checkLValue(scope, lvalue, assignee, transform);
|
2023-11-17 18:15:31 +00:00
|
|
|
assignees.push_back(assignee);
|
2024-01-19 15:13:08 +00:00
|
|
|
++i;
|
2023-10-13 20:38:31 +01:00
|
|
|
}
|
|
|
|
|
2023-11-17 18:15:31 +00:00
|
|
|
TypePackId resultPack = checkPack(scope, assign->values).tp;
|
|
|
|
addConstraint(scope, assign->location, UnpackConstraint{arena->addTypePack(std::move(assignees)), resultPack, /*resultIsLValue*/ true});
|
2023-03-17 14:59:30 +00:00
|
|
|
|
|
|
|
return ControlFlow::None;
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatCompoundAssign* assign)
|
2022-08-18 22:04:33 +01:00
|
|
|
{
|
2023-10-20 21:36:26 +01:00
|
|
|
AstExprBinary binop = AstExprBinary{assign->location, assign->op, assign->var, assign->value};
|
|
|
|
TypeId resultTy = check(scope, &binop).ty;
|
2023-11-03 19:47:28 +00:00
|
|
|
|
2024-01-19 15:13:08 +00:00
|
|
|
checkLValue(scope, assign->var, resultTy, true);
|
2022-08-18 22:04:33 +01:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
DefId def = dfg->getDef(assign->var);
|
|
|
|
scope->lvalueTypes[def] = resultTy;
|
2023-03-17 14:59:30 +00:00
|
|
|
|
|
|
|
return ControlFlow::None;
|
2022-08-18 22:04:33 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatIf* ifStatement)
|
2022-06-17 01:54:42 +01:00
|
|
|
{
|
2023-10-06 18:31:16 +01:00
|
|
|
RefinementId refinement = check(scope, ifStatement->condition, std::nullopt).refinement;
|
2022-06-17 01:54:42 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
ScopePtr thenScope = childScope(ifStatement->thenbody, scope);
|
2023-02-10 18:50:54 +00:00
|
|
|
applyRefinements(thenScope, ifStatement->condition->location, refinement);
|
2022-06-17 01:54:42 +01:00
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
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;
|
2022-06-17 01:54:42 +01:00
|
|
|
if (ifStatement->elsebody)
|
2023-03-17 14:59:30 +00:00
|
|
|
elsecf = visit(elseScope, ifStatement->elsebody);
|
|
|
|
|
2023-09-30 01:22:06 +01:00
|
|
|
if (thencf != ControlFlow::None && elsecf == ControlFlow::None)
|
2023-03-17 14:59:30 +00:00
|
|
|
scope->inheritRefinements(elseScope);
|
2023-09-30 01:22:06 +01:00
|
|
|
else if (thencf == ControlFlow::None && elsecf != ControlFlow::None)
|
2023-03-17 14:59:30 +00:00
|
|
|
scope->inheritRefinements(thenScope);
|
|
|
|
|
2023-11-10 18:05:48 +00:00
|
|
|
if (thencf == ControlFlow::None)
|
|
|
|
scope->inheritAssignments(thenScope);
|
|
|
|
if (elsecf == ControlFlow::None)
|
|
|
|
scope->inheritAssignments(elseScope);
|
|
|
|
|
2023-09-30 01:22:06 +01:00
|
|
|
if (FFlag::LuauLoopControlFlowAnalysis && thencf == elsecf)
|
|
|
|
return thencf;
|
|
|
|
else if (matches(thencf, ControlFlow::Returns | ControlFlow::Throws) && matches(elsecf, ControlFlow::Returns | ControlFlow::Throws))
|
2023-03-17 14:59:30 +00:00
|
|
|
return ControlFlow::Returns;
|
|
|
|
else
|
|
|
|
return ControlFlow::None;
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
|
2023-02-17 14:53:37 +00:00
|
|
|
static bool occursCheck(TypeId needle, TypeId haystack)
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(get<BlockedType>(needle));
|
|
|
|
haystack = follow(haystack);
|
|
|
|
|
|
|
|
auto checkHaystack = [needle](TypeId haystack) {
|
|
|
|
return occursCheck(needle, haystack);
|
|
|
|
};
|
|
|
|
|
|
|
|
if (needle == haystack)
|
|
|
|
return true;
|
|
|
|
else if (auto ut = get<UnionType>(haystack))
|
|
|
|
return std::any_of(begin(ut), end(ut), checkHaystack);
|
|
|
|
else if (auto it = get<IntersectionType>(haystack))
|
|
|
|
return std::any_of(begin(it), end(it), checkHaystack);
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatTypeAlias* alias)
|
2022-06-24 02:44:07 +01:00
|
|
|
{
|
2024-01-12 19:16:39 +00:00
|
|
|
if (alias->name == kParseNameError)
|
|
|
|
return ControlFlow::None;
|
|
|
|
|
|
|
|
if (alias->name == "typeof")
|
|
|
|
{
|
|
|
|
reportError(alias->location, GenericError{"Type aliases cannot be named typeof"});
|
|
|
|
return ControlFlow::None;
|
|
|
|
}
|
|
|
|
|
2023-02-17 14:53:37 +00:00
|
|
|
ScopePtr* defnScope = astTypeAliasDefiningScopes.find(alias);
|
|
|
|
|
|
|
|
std::unordered_map<Name, TypeFun>* typeBindings;
|
|
|
|
if (alias->exported)
|
|
|
|
typeBindings = &scope->exportedTypeBindings;
|
|
|
|
else
|
|
|
|
typeBindings = &scope->privateTypeBindings;
|
|
|
|
|
2022-08-04 22:27:28 +01:00
|
|
|
// These will be undefined if the alias was a duplicate definition, in which
|
|
|
|
// case we just skip over it.
|
2023-02-17 14:53:37 +00:00
|
|
|
auto bindingIt = typeBindings->find(alias->name.value);
|
|
|
|
if (bindingIt == typeBindings->end() || defnScope == nullptr)
|
2023-03-17 14:59:30 +00:00
|
|
|
return ControlFlow::None;
|
2022-06-24 02:44:07 +01:00
|
|
|
|
2023-02-17 14:53:37 +00:00
|
|
|
TypeId ty = resolveType(*defnScope, alias->type, /* inTypeArguments */ false);
|
2022-08-04 22:27:28 +01:00
|
|
|
|
2023-02-17 14:53:37 +00:00
|
|
|
TypeId aliasTy = bindingIt->second.type;
|
|
|
|
LUAU_ASSERT(get<BlockedType>(aliasTy));
|
|
|
|
|
|
|
|
if (occursCheck(aliasTy, ty))
|
2022-09-02 00:00:14 +01:00
|
|
|
{
|
2023-02-17 14:53:37 +00:00
|
|
|
asMutable(aliasTy)->ty.emplace<BoundType>(builtinTypes->anyType);
|
|
|
|
reportError(alias->nameLocation, OccursCheckFailed{});
|
2022-09-02 00:00:14 +01:00
|
|
|
}
|
2023-02-17 14:53:37 +00:00
|
|
|
else
|
|
|
|
asMutable(aliasTy)->ty.emplace<BoundType>(ty);
|
|
|
|
|
|
|
|
std::vector<TypeId> typeParams;
|
2023-02-24 18:24:22 +00:00
|
|
|
for (auto tyParam : createGenerics(*defnScope, alias->generics, /* useCache */ true, /* addTypes */ false))
|
2023-02-17 14:53:37 +00:00
|
|
|
typeParams.push_back(tyParam.second.ty);
|
|
|
|
|
|
|
|
std::vector<TypePackId> typePackParams;
|
2023-02-24 18:24:22 +00:00
|
|
|
for (auto tpParam : createGenericPacks(*defnScope, alias->genericPacks, /* useCache */ true, /* addTypes */ false))
|
2023-02-17 14:53:37 +00:00
|
|
|
typePackParams.push_back(tpParam.second.tp);
|
|
|
|
|
|
|
|
addConstraint(scope, alias->type->location,
|
|
|
|
NameConstraint{
|
|
|
|
ty,
|
|
|
|
alias->name.value,
|
|
|
|
/*synthetic=*/false,
|
|
|
|
std::move(typeParams),
|
|
|
|
std::move(typePackParams),
|
|
|
|
});
|
2023-03-17 14:59:30 +00:00
|
|
|
|
|
|
|
return ControlFlow::None;
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatDeclareGlobal* global)
|
2022-08-04 22:27:28 +01:00
|
|
|
{
|
|
|
|
LUAU_ASSERT(global->type);
|
|
|
|
|
2023-01-20 12:02:39 +00:00
|
|
|
TypeId globalTy = resolveType(scope, global->type, /* inTypeArguments */ false);
|
2022-08-11 21:42:54 +01:00
|
|
|
Name globalName(global->name.value);
|
|
|
|
|
|
|
|
module->declaredGlobals[globalName] = globalTy;
|
2023-03-03 13:45:38 +00:00
|
|
|
rootScope->bindings[global->name] = Binding{globalTy, global->location};
|
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
DefId def = dfg->getDef(global);
|
|
|
|
rootScope->lvalueTypes[def] = globalTy;
|
|
|
|
rootScope->rvalueRefinements[def] = globalTy;
|
2023-03-17 14:59:30 +00:00
|
|
|
|
|
|
|
return ControlFlow::None;
|
2022-08-04 22:27:28 +01:00
|
|
|
}
|
|
|
|
|
2022-08-11 21:42:54 +01:00
|
|
|
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" ||
|
2023-09-01 17:38:53 +01:00
|
|
|
name == "__metatable" || name == "__eq" || name == "__lt" || name == "__le" || name == "__mode" || name == "__iter" || name == "__len" ||
|
2023-11-10 18:05:48 +00:00
|
|
|
name == "__idiv";
|
2022-08-11 21:42:54 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatDeclareClass* declaredClass)
|
2022-08-04 22:27:28 +01:00
|
|
|
{
|
2023-05-05 20:57:12 +01:00
|
|
|
std::optional<TypeId> superTy = std::make_optional(builtinTypes->classType);
|
2022-08-11 21:42:54 +01:00
|
|
|
if (declaredClass->superName)
|
|
|
|
{
|
|
|
|
Name superName = Name(declaredClass->superName->value);
|
|
|
|
std::optional<TypeFun> lookupType = scope->lookupType(superName);
|
|
|
|
|
|
|
|
if (!lookupType)
|
|
|
|
{
|
|
|
|
reportError(declaredClass->location, UnknownSymbol{superName, UnknownSymbol::Type});
|
2023-03-17 14:59:30 +00:00
|
|
|
return ControlFlow::None;
|
2022-08-11 21:42:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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;
|
|
|
|
|
2023-01-03 17:33:19 +00:00
|
|
|
if (!get<ClassType>(follow(*superTy)))
|
2022-08-11 21:42:54 +01:00
|
|
|
{
|
|
|
|
reportError(declaredClass->location,
|
|
|
|
GenericError{format("Cannot use non-class type '%s' as a superclass of class '%s'", superName.c_str(), declaredClass->name.value)});
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
return ControlFlow::None;
|
2022-08-11 21:42:54 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Name className(declaredClass->name.value);
|
|
|
|
|
2023-04-21 22:41:03 +01:00
|
|
|
TypeId classTy = arena->addType(ClassType(className, {}, superTy, std::nullopt, {}, {}, module->name));
|
2023-01-03 17:33:19 +00:00
|
|
|
ClassType* ctv = getMutable<ClassType>(classTy);
|
2022-08-11 21:42:54 +01:00
|
|
|
|
2023-01-03 17:33:19 +00:00
|
|
|
TypeId metaTy = arena->addType(TableType{TableState::Sealed, scope->level, scope.get()});
|
|
|
|
TableType* metatable = getMutable<TableType>(metaTy);
|
2022-08-11 21:42:54 +01:00
|
|
|
|
|
|
|
ctv->metatable = metaTy;
|
|
|
|
|
|
|
|
scope->exportedTypeBindings[className] = TypeFun{{}, classTy};
|
|
|
|
|
2023-11-10 18:05:48 +00:00
|
|
|
if (declaredClass->indexer)
|
2023-06-16 18:01:18 +01:00
|
|
|
{
|
|
|
|
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),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-11 21:42:54 +01:00
|
|
|
for (const AstDeclaredClassProp& prop : declaredClass->props)
|
|
|
|
{
|
|
|
|
Name propName(prop.name.value);
|
2023-01-20 12:02:39 +00:00
|
|
|
TypeId propTy = resolveType(scope, prop.ty, /* inTypeArguments */ false);
|
2022-08-11 21:42:54 +01:00
|
|
|
|
|
|
|
bool assignToMetatable = isMetamethod(propName);
|
|
|
|
|
|
|
|
// Function types always take 'self', but this isn't reflected in the
|
|
|
|
// parsed annotation. Add it here.
|
|
|
|
if (prop.isMethod)
|
|
|
|
{
|
2023-01-03 17:33:19 +00:00
|
|
|
if (FunctionType* ftv = getMutable<FunctionType>(propTy))
|
2022-08-11 21:42:54 +01:00
|
|
|
{
|
|
|
|
ftv->argNames.insert(ftv->argNames.begin(), FunctionArgument{"self", {}});
|
2024-02-02 18:20:03 +00:00
|
|
|
ftv->argTypes = addTypePack({classTy}, ftv->argTypes);
|
2022-08-11 21:42:54 +01:00
|
|
|
|
|
|
|
ftv->hasSelf = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ctv->props.count(propName) == 0)
|
|
|
|
{
|
|
|
|
if (assignToMetatable)
|
|
|
|
metatable->props[propName] = {propTy};
|
|
|
|
else
|
|
|
|
ctv->props[propName] = {propTy};
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-04-28 12:55:55 +01:00
|
|
|
TypeId currentTy = assignToMetatable ? metatable->props[propName].type() : ctv->props[propName].type();
|
2022-08-11 21:42:54 +01:00
|
|
|
|
|
|
|
// We special-case this logic to keep the intersection flat; otherwise we
|
|
|
|
// would create a ton of nested intersection types.
|
2023-01-03 17:33:19 +00:00
|
|
|
if (const IntersectionType* itv = get<IntersectionType>(currentTy))
|
2022-08-11 21:42:54 +01:00
|
|
|
{
|
|
|
|
std::vector<TypeId> options = itv->parts;
|
|
|
|
options.push_back(propTy);
|
2023-01-03 17:33:19 +00:00
|
|
|
TypeId newItv = arena->addType(IntersectionType{std::move(options)});
|
2022-08-11 21:42:54 +01:00
|
|
|
|
|
|
|
if (assignToMetatable)
|
|
|
|
metatable->props[propName] = {newItv};
|
|
|
|
else
|
|
|
|
ctv->props[propName] = {newItv};
|
|
|
|
}
|
2023-01-03 17:33:19 +00:00
|
|
|
else if (get<FunctionType>(currentTy))
|
2022-08-11 21:42:54 +01:00
|
|
|
{
|
2023-01-03 17:33:19 +00:00
|
|
|
TypeId intersection = arena->addType(IntersectionType{{currentTy, propTy}});
|
2022-08-11 21:42:54 +01:00
|
|
|
|
|
|
|
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())});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-03-17 14:59:30 +00:00
|
|
|
|
|
|
|
return ControlFlow::None;
|
2022-08-04 22:27:28 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatDeclareFunction* global)
|
2022-08-04 22:27:28 +01:00
|
|
|
{
|
2022-08-11 21:42:54 +01:00
|
|
|
std::vector<std::pair<Name, GenericTypeDefinition>> generics = createGenerics(scope, global->generics);
|
|
|
|
std::vector<std::pair<Name, GenericTypePackDefinition>> genericPacks = createGenericPacks(scope, global->genericPacks);
|
|
|
|
|
|
|
|
std::vector<TypeId> genericTys;
|
|
|
|
genericTys.reserve(generics.size());
|
|
|
|
for (auto& [name, generic] : generics)
|
2022-09-08 22:44:50 +01:00
|
|
|
{
|
2022-08-11 21:42:54 +01:00
|
|
|
genericTys.push_back(generic.ty);
|
2022-09-08 22:44:50 +01:00
|
|
|
}
|
2022-08-11 21:42:54 +01:00
|
|
|
|
|
|
|
std::vector<TypePackId> genericTps;
|
|
|
|
genericTps.reserve(genericPacks.size());
|
|
|
|
for (auto& [name, generic] : genericPacks)
|
2022-09-08 22:44:50 +01:00
|
|
|
{
|
2022-08-11 21:42:54 +01:00
|
|
|
genericTps.push_back(generic.tp);
|
2022-09-08 22:44:50 +01:00
|
|
|
}
|
2022-08-11 21:42:54 +01:00
|
|
|
|
|
|
|
ScopePtr funScope = scope;
|
|
|
|
if (!generics.empty() || !genericPacks.empty())
|
2022-08-18 22:04:33 +01:00
|
|
|
funScope = childScope(global, scope);
|
2022-08-11 21:42:54 +01:00
|
|
|
|
2023-01-20 12:02:39 +00:00
|
|
|
TypePackId paramPack = resolveTypePack(funScope, global->params, /* inTypeArguments */ false);
|
|
|
|
TypePackId retPack = resolveTypePack(funScope, global->retTypes, /* inTypeArguments */ false);
|
2023-01-03 17:33:19 +00:00
|
|
|
TypeId fnType = arena->addType(FunctionType{TypeLevel{}, funScope.get(), std::move(genericTys), std::move(genericTps), paramPack, retPack});
|
|
|
|
FunctionType* ftv = getMutable<FunctionType>(fnType);
|
2023-10-06 18:31:16 +01:00
|
|
|
ftv->isCheckedFunction = global->checkedFunction;
|
2022-08-11 21:42:54 +01:00
|
|
|
|
|
|
|
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};
|
2023-03-03 13:45:38 +00:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
DefId def = dfg->getDef(global);
|
|
|
|
rootScope->lvalueTypes[def] = fnType;
|
|
|
|
rootScope->rvalueRefinements[def] = fnType;
|
2023-03-17 14:59:30 +00:00
|
|
|
|
|
|
|
return ControlFlow::None;
|
2022-08-04 22:27:28 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatError* error)
|
2022-09-29 23:11:54 +01:00
|
|
|
{
|
|
|
|
for (AstStat* stat : error->statements)
|
|
|
|
visit(scope, stat);
|
|
|
|
for (AstExpr* expr : error->expressions)
|
|
|
|
check(scope, expr);
|
2023-03-17 14:59:30 +00:00
|
|
|
|
|
|
|
return ControlFlow::None;
|
2022-09-29 23:11:54 +01:00
|
|
|
}
|
|
|
|
|
2024-02-16 01:25:31 +00:00
|
|
|
InferencePack ConstraintGenerator::checkPack(const ScopePtr& scope, AstArray<AstExpr*> exprs, const std::vector<std::optional<TypeId>>& expectedTypes)
|
2022-06-03 21:32:20 +01:00
|
|
|
{
|
2022-09-02 00:00:14 +01:00
|
|
|
std::vector<TypeId> head;
|
|
|
|
std::optional<TypePackId> tail;
|
2022-06-17 01:54:42 +01:00
|
|
|
|
|
|
|
for (size_t i = 0; i < exprs.size; ++i)
|
|
|
|
{
|
|
|
|
AstExpr* expr = exprs.data[i];
|
|
|
|
if (i < exprs.size - 1)
|
2022-09-23 19:32:10 +01:00
|
|
|
{
|
|
|
|
std::optional<TypeId> expectedType;
|
|
|
|
if (i < expectedTypes.size())
|
|
|
|
expectedType = expectedTypes[i];
|
2023-10-06 18:31:16 +01:00
|
|
|
head.push_back(check(scope, expr, expectedType).ty);
|
2022-09-23 19:32:10 +01:00
|
|
|
}
|
2022-06-17 01:54:42 +01:00
|
|
|
else
|
2022-09-23 19:32:10 +01:00
|
|
|
{
|
2023-02-17 14:53:37 +00:00
|
|
|
std::vector<std::optional<TypeId>> expectedTailTypes;
|
2022-09-29 23:11:54 +01:00
|
|
|
if (i < expectedTypes.size())
|
|
|
|
expectedTailTypes.assign(begin(expectedTypes) + i, end(expectedTypes));
|
2022-10-27 23:22:49 +01:00
|
|
|
tail = checkPack(scope, expr, expectedTailTypes).tp;
|
2022-09-23 19:32:10 +01:00
|
|
|
}
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
|
2024-02-02 18:20:03 +00:00
|
|
|
return InferencePack{addTypePack(std::move(head), tail)};
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
InferencePack ConstraintGenerator::checkPack(
|
2023-09-30 01:22:06 +01:00
|
|
|
const ScopePtr& scope, AstExpr* expr, const std::vector<std::optional<TypeId>>& expectedTypes, bool generalize)
|
2022-06-03 21:32:20 +01:00
|
|
|
{
|
2022-07-01 00:29:02 +01:00
|
|
|
RecursionCounter counter{&recursionCount};
|
|
|
|
|
|
|
|
if (recursionCount >= FInt::LuauCheckRecursionLimit)
|
|
|
|
{
|
|
|
|
reportCodeTooComplex(expr->location);
|
2023-01-03 17:33:19 +00:00
|
|
|
return InferencePack{builtinTypes->errorRecoveryTypePack()};
|
2022-07-01 00:29:02 +01:00
|
|
|
}
|
2022-06-03 21:32:20 +01:00
|
|
|
|
2022-10-27 23:22:49 +01:00
|
|
|
InferencePack result;
|
2022-06-17 01:54:42 +01:00
|
|
|
|
|
|
|
if (AstExprCall* call = expr->as<AstExprCall>())
|
2023-02-17 14:53:37 +00:00
|
|
|
result = checkPack(scope, call);
|
2022-07-01 00:29:02 +01:00
|
|
|
else if (AstExprVarargs* varargs = expr->as<AstExprVarargs>())
|
|
|
|
{
|
|
|
|
if (scope->varargPack)
|
2022-10-27 23:22:49 +01:00
|
|
|
result = InferencePack{*scope->varargPack};
|
2022-07-01 00:29:02 +01:00
|
|
|
else
|
2023-01-03 17:33:19 +00:00
|
|
|
result = InferencePack{builtinTypes->errorRecoveryTypePack()};
|
2022-07-01 00:29:02 +01:00
|
|
|
}
|
2022-06-17 01:54:42 +01:00
|
|
|
else
|
|
|
|
{
|
2022-09-23 19:32:10 +01:00
|
|
|
std::optional<TypeId> expectedType;
|
|
|
|
if (!expectedTypes.empty())
|
|
|
|
expectedType = expectedTypes[0];
|
2023-10-06 18:31:16 +01:00
|
|
|
TypeId t = check(scope, expr, expectedType, /*forceSingletons*/ false, generalize).ty;
|
2022-10-27 23:22:49 +01:00
|
|
|
result = InferencePack{arena->addTypePack({t})};
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
|
2022-10-27 23:22:49 +01:00
|
|
|
LUAU_ASSERT(result.tp);
|
2023-02-03 12:34:12 +00:00
|
|
|
module->astTypePacks[expr] = result.tp;
|
2022-06-17 01:54:42 +01:00
|
|
|
return result;
|
2022-06-03 21:32:20 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
InferencePack ConstraintGenerator::checkPack(const ScopePtr& scope, AstExprCall* call)
|
2022-06-03 21:32:20 +01:00
|
|
|
{
|
2022-12-02 10:46:05 +00:00
|
|
|
std::vector<AstExpr*> exprArgs;
|
2023-02-10 18:50:54 +00:00
|
|
|
|
|
|
|
std::vector<RefinementId> returnRefinements;
|
|
|
|
std::vector<std::optional<TypeId>> discriminantTypes;
|
|
|
|
|
2022-12-02 10:46:05 +00:00
|
|
|
if (call->self)
|
|
|
|
{
|
|
|
|
AstExprIndexName* indexExpr = call->func->as<AstExprIndexName>();
|
|
|
|
if (!indexExpr)
|
|
|
|
ice->ice("method call expression has no 'self'");
|
|
|
|
|
|
|
|
exprArgs.push_back(indexExpr->expr);
|
2023-02-10 18:50:54 +00:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
if (auto key = dfg->getRefinementKey(indexExpr->expr))
|
2023-02-10 18:50:54 +00:00
|
|
|
{
|
|
|
|
TypeId discriminantTy = arena->addType(BlockedType{});
|
2023-10-20 21:36:26 +01:00
|
|
|
returnRefinements.push_back(refinementArena.proposition(key, discriminantTy));
|
2023-02-10 18:50:54 +00:00
|
|
|
discriminantTypes.push_back(discriminantTy);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
discriminantTypes.push_back(std::nullopt);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (AstExpr* arg : call->args)
|
|
|
|
{
|
|
|
|
exprArgs.push_back(arg);
|
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
if (auto key = dfg->getRefinementKey(arg))
|
2023-02-10 18:50:54 +00:00
|
|
|
{
|
|
|
|
TypeId discriminantTy = arena->addType(BlockedType{});
|
2023-10-20 21:36:26 +01:00
|
|
|
returnRefinements.push_back(refinementArena.proposition(key, discriminantTy));
|
2023-02-10 18:50:54 +00:00
|
|
|
discriminantTypes.push_back(discriminantTy);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
discriminantTypes.push_back(std::nullopt);
|
2022-12-02 10:46:05 +00:00
|
|
|
}
|
|
|
|
|
2024-01-27 02:30:40 +00:00
|
|
|
Checkpoint funcBeginCheckpoint = checkpoint(this);
|
|
|
|
|
2022-10-27 23:22:49 +01:00
|
|
|
TypeId fnType = check(scope, call->func).ty;
|
2022-07-01 00:29:02 +01:00
|
|
|
|
2024-01-27 02:30:40 +00:00
|
|
|
Checkpoint funcEndCheckpoint = checkpoint(this);
|
|
|
|
|
2023-03-03 13:45:38 +00:00
|
|
|
std::vector<std::optional<TypeId>> expectedTypesForCall = getExpectedCallTypesForFunctionOverloads(fnType);
|
|
|
|
|
2023-02-10 18:50:54 +00:00
|
|
|
module->astOriginalCallTypes[call->func] = fnType;
|
2023-07-07 18:14:35 +01:00
|
|
|
module->astOriginalCallTypes[call] = fnType;
|
2023-02-10 18:50:54 +00:00
|
|
|
|
2023-09-30 01:22:06 +01:00
|
|
|
Checkpoint argBeginCheckpoint = checkpoint(this);
|
2022-10-27 23:22:49 +01:00
|
|
|
|
2022-12-02 10:46:05 +00:00
|
|
|
std::vector<TypeId> args;
|
|
|
|
std::optional<TypePackId> argTail;
|
2023-02-03 12:34:12 +00:00
|
|
|
std::vector<RefinementId> argumentRefinements;
|
2022-11-18 18:45:14 +00:00
|
|
|
|
2022-12-02 10:46:05 +00:00
|
|
|
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.
|
2023-02-03 12:34:12 +00:00
|
|
|
TypeId* selfTy = module->astTypes.find(exprArgs[0]);
|
2022-12-02 10:46:05 +00:00
|
|
|
if (selfTy)
|
|
|
|
args.push_back(*selfTy);
|
|
|
|
else
|
2023-08-04 18:01:35 +01:00
|
|
|
args.push_back(freshType(scope));
|
2022-12-02 10:46:05 +00:00
|
|
|
}
|
|
|
|
else if (i < exprArgs.size() - 1 || !(arg->is<AstExprCall>() || arg->is<AstExprVarargs>()))
|
2022-12-09 18:07:25 +00:00
|
|
|
{
|
2023-10-06 18:31:16 +01:00
|
|
|
auto [ty, refinement] = check(scope, arg, /*expectedType*/ std::nullopt, /*forceSingleton*/ false, /*generalize*/ false);
|
2022-12-09 18:07:25 +00:00
|
|
|
args.push_back(ty);
|
2023-02-03 12:34:12 +00:00
|
|
|
argumentRefinements.push_back(refinement);
|
2022-12-09 18:07:25 +00:00
|
|
|
}
|
2022-12-02 10:46:05 +00:00
|
|
|
else
|
2023-02-10 18:50:54 +00:00
|
|
|
{
|
2023-02-17 14:53:37 +00:00
|
|
|
auto [tp, refis] = checkPack(scope, arg, {});
|
2023-02-10 18:50:54 +00:00
|
|
|
argTail = tp;
|
|
|
|
argumentRefinements.insert(argumentRefinements.end(), refis.begin(), refis.end());
|
|
|
|
}
|
2022-11-18 18:45:14 +00:00
|
|
|
}
|
2022-06-17 01:54:42 +01:00
|
|
|
|
2022-12-02 10:46:05 +00:00
|
|
|
Checkpoint argEndCheckpoint = checkpoint(this);
|
|
|
|
|
2022-10-27 23:22:49 +01:00
|
|
|
if (matchSetmetatable(*call))
|
2022-09-23 19:32:10 +01:00
|
|
|
{
|
2022-12-02 10:46:05 +00:00
|
|
|
TypePack argTailPack;
|
|
|
|
if (argTail && args.size() < 2)
|
2023-01-03 17:33:19 +00:00
|
|
|
argTailPack = extendTypePack(*arena, builtinTypes, *argTail, 2 - args.size());
|
2022-12-02 10:46:05 +00:00
|
|
|
|
2023-03-24 17:34:14 +00:00
|
|
|
TypeId target = nullptr;
|
|
|
|
TypeId mt = nullptr;
|
2022-12-02 10:46:05 +00:00
|
|
|
|
2023-03-24 17:34:14 +00:00
|
|
|
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<TypeId> 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);
|
2022-10-27 23:22:49 +01:00
|
|
|
|
2024-01-12 19:16:39 +00:00
|
|
|
target = follow(target);
|
|
|
|
|
2022-10-27 23:22:49 +01:00
|
|
|
AstExpr* targetExpr = call->args.data[0];
|
|
|
|
|
2024-01-12 19:16:39 +00:00
|
|
|
TypeId resultTy = nullptr;
|
|
|
|
|
|
|
|
if (isTableUnion(target))
|
|
|
|
{
|
|
|
|
const UnionType* targetUnion = get<UnionType>(target);
|
|
|
|
std::vector<TypeId> 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});
|
2022-10-27 23:22:49 +01:00
|
|
|
|
|
|
|
if (AstExprLocal* targetLocal = targetExpr->as<AstExprLocal>())
|
2023-01-27 21:28:45 +00:00
|
|
|
{
|
2022-10-27 23:22:49 +01:00
|
|
|
scope->bindings[targetLocal->local].typeId = resultTy;
|
2023-03-03 13:45:38 +00:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
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
|
|
|
|
|
2023-11-17 18:15:31 +00:00
|
|
|
recordInferredBinding(targetLocal->local, resultTy);
|
2023-01-27 21:28:45 +00:00
|
|
|
}
|
|
|
|
|
2023-02-10 18:50:54 +00:00
|
|
|
return InferencePack{arena->addTypePack({resultTy}), {refinementArena.variadic(returnRefinements)}};
|
2022-09-23 19:32:10 +01:00
|
|
|
}
|
2022-10-27 23:22:49 +01:00
|
|
|
else
|
2022-09-23 19:32:10 +01:00
|
|
|
{
|
2023-02-10 18:50:54 +00:00
|
|
|
if (matchAssert(*call) && !argumentRefinements.empty())
|
|
|
|
applyRefinements(scope, call->args.data[0]->location, argumentRefinements[0]);
|
2022-10-27 23:22:49 +01:00
|
|
|
|
|
|
|
// TODO: How do expectedTypes play into this? Do they?
|
|
|
|
TypePackId rets = arena->addTypePack(BlockedTypePack{});
|
2024-02-02 18:20:03 +00:00
|
|
|
TypePackId argPack = addTypePack(std::move(args), argTail);
|
2023-02-24 18:24:22 +00:00
|
|
|
FunctionType ftv(TypeLevel{}, scope.get(), argPack, rets, std::nullopt, call->self);
|
2022-10-27 23:22:49 +01:00
|
|
|
|
2024-01-27 02:30:40 +00:00
|
|
|
/*
|
|
|
|
* To make bidirectional type checking work, we need to solve these constraints in a particular order:
|
|
|
|
*
|
|
|
|
* 1. Solve the function type
|
|
|
|
* 2. Propagate type information from the function type to the argument types
|
|
|
|
* 3. Solve the argument types
|
|
|
|
* 4. Solve the call
|
|
|
|
*/
|
|
|
|
|
2024-02-16 01:25:31 +00:00
|
|
|
NotNull<Constraint> checkConstraint =
|
|
|
|
addConstraint(scope, call->func->location, FunctionCheckConstraint{fnType, argPack, call, NotNull{&module->astExpectedTypes}});
|
2024-02-09 17:32:52 +00:00
|
|
|
|
|
|
|
forEachConstraint(funcBeginCheckpoint, funcEndCheckpoint, this, [checkConstraint](const ConstraintPtr& constraint) {
|
|
|
|
checkConstraint->dependencies.emplace_back(constraint.get());
|
2024-01-27 02:30:40 +00:00
|
|
|
});
|
|
|
|
|
2024-02-09 17:32:52 +00:00
|
|
|
NotNull<Constraint> callConstraint = addConstraint(scope, call->func->location,
|
|
|
|
FunctionCallConstraint{
|
|
|
|
fnType,
|
|
|
|
argPack,
|
|
|
|
rets,
|
|
|
|
call,
|
|
|
|
std::move(discriminantTypes),
|
|
|
|
&module->astOverloadResolvedTypes,
|
|
|
|
});
|
|
|
|
|
|
|
|
callConstraint->dependencies.push_back(checkConstraint);
|
|
|
|
|
|
|
|
forEachConstraint(argBeginCheckpoint, argEndCheckpoint, this, [checkConstraint, callConstraint](const ConstraintPtr& constraint) {
|
|
|
|
constraint->dependencies.emplace_back(checkConstraint);
|
2024-01-27 02:30:40 +00:00
|
|
|
|
2024-02-09 17:32:52 +00:00
|
|
|
callConstraint->dependencies.emplace_back(constraint.get());
|
2022-12-02 10:46:05 +00:00
|
|
|
});
|
|
|
|
|
2023-02-10 18:50:54 +00:00
|
|
|
return InferencePack{rets, {refinementArena.variadic(returnRefinements)}};
|
2022-09-23 19:32:10 +01:00
|
|
|
}
|
2022-10-27 23:22:49 +01:00
|
|
|
}
|
|
|
|
|
2024-02-16 01:25:31 +00:00
|
|
|
Inference ConstraintGenerator::check(const ScopePtr& scope, AstExpr* expr, std::optional<TypeId> expectedType, bool forceSingleton, bool generalize)
|
2022-10-27 23:22:49 +01:00
|
|
|
{
|
|
|
|
RecursionCounter counter{&recursionCount};
|
|
|
|
|
|
|
|
if (recursionCount >= FInt::LuauCheckRecursionLimit)
|
|
|
|
{
|
|
|
|
reportCodeTooComplex(expr->location);
|
2023-01-03 17:33:19 +00:00
|
|
|
return Inference{builtinTypes->errorRecoveryType()};
|
2022-10-27 23:22:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Inference result;
|
|
|
|
|
|
|
|
if (auto group = expr->as<AstExprGroup>())
|
2023-10-06 18:31:16 +01:00
|
|
|
result = check(scope, group->expr, expectedType, forceSingleton);
|
2022-10-27 23:22:49 +01:00
|
|
|
else if (auto stringExpr = expr->as<AstExprConstantString>())
|
2022-11-04 17:02:37 +00:00
|
|
|
result = check(scope, stringExpr, expectedType, forceSingleton);
|
2022-10-27 23:22:49 +01:00
|
|
|
else if (expr->is<AstExprConstantNumber>())
|
2023-01-03 17:33:19 +00:00
|
|
|
result = Inference{builtinTypes->numberType};
|
2022-10-27 23:22:49 +01:00
|
|
|
else if (auto boolExpr = expr->as<AstExprConstantBool>())
|
2022-11-04 17:02:37 +00:00
|
|
|
result = check(scope, boolExpr, expectedType, forceSingleton);
|
2022-06-17 01:54:42 +01:00
|
|
|
else if (expr->is<AstExprConstantNil>())
|
2023-01-03 17:33:19 +00:00
|
|
|
result = Inference{builtinTypes->nilType};
|
2022-10-21 18:33:43 +01:00
|
|
|
else if (auto local = expr->as<AstExprLocal>())
|
2023-10-06 18:31:16 +01:00
|
|
|
result = check(scope, local);
|
2022-10-21 18:33:43 +01:00
|
|
|
else if (auto global = expr->as<AstExprGlobal>())
|
|
|
|
result = check(scope, global);
|
2022-07-01 00:29:02 +01:00
|
|
|
else if (expr->is<AstExprVarargs>())
|
|
|
|
result = flattenPack(scope, expr->location, checkPack(scope, expr));
|
2022-10-27 23:22:49 +01:00
|
|
|
else if (auto call = expr->as<AstExprCall>())
|
2023-02-17 14:53:37 +00:00
|
|
|
result = flattenPack(scope, expr->location, checkPack(scope, call)); // TODO: needs predicates too
|
2022-06-17 01:54:42 +01:00
|
|
|
else if (auto a = expr->as<AstExprFunction>())
|
2023-09-30 01:22:06 +01:00
|
|
|
result = check(scope, a, expectedType, generalize);
|
2022-06-17 01:54:42 +01:00
|
|
|
else if (auto indexName = expr->as<AstExprIndexName>())
|
|
|
|
result = check(scope, indexName);
|
2022-07-01 00:29:02 +01:00
|
|
|
else if (auto indexExpr = expr->as<AstExprIndexExpr>())
|
|
|
|
result = check(scope, indexExpr);
|
2022-06-17 01:54:42 +01:00
|
|
|
else if (auto table = expr->as<AstExprTable>())
|
2022-09-23 19:32:10 +01:00
|
|
|
result = check(scope, table, expectedType);
|
2022-07-01 00:29:02 +01:00
|
|
|
else if (auto unary = expr->as<AstExprUnary>())
|
|
|
|
result = check(scope, unary);
|
|
|
|
else if (auto binary = expr->as<AstExprBinary>())
|
2022-10-21 18:33:43 +01:00
|
|
|
result = check(scope, binary, expectedType);
|
2022-08-18 22:04:33 +01:00
|
|
|
else if (auto ifElse = expr->as<AstExprIfElse>())
|
2022-09-23 19:32:10 +01:00
|
|
|
result = check(scope, ifElse, expectedType);
|
2022-08-11 21:42:54 +01:00
|
|
|
else if (auto typeAssert = expr->as<AstExprTypeAssertion>())
|
|
|
|
result = check(scope, typeAssert);
|
2023-01-27 21:28:45 +00:00
|
|
|
else if (auto interpString = expr->as<AstExprInterpString>())
|
|
|
|
result = check(scope, interpString);
|
2022-07-01 00:29:02 +01:00
|
|
|
else if (auto err = expr->as<AstExprError>())
|
|
|
|
{
|
|
|
|
// Open question: Should we traverse into this?
|
2022-09-29 23:11:54 +01:00
|
|
|
for (AstExpr* subExpr : err->expressions)
|
|
|
|
check(scope, subExpr);
|
|
|
|
|
2023-01-03 17:33:19 +00:00
|
|
|
result = Inference{builtinTypes->errorRecoveryType()};
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(0);
|
2022-10-27 23:22:49 +01:00
|
|
|
result = Inference{freshType(scope)};
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
|
2022-10-27 23:22:49 +01:00
|
|
|
LUAU_ASSERT(result.ty);
|
2023-02-03 12:34:12 +00:00
|
|
|
module->astTypes[expr] = result.ty;
|
2023-01-27 21:28:45 +00:00
|
|
|
if (expectedType)
|
2023-02-03 12:34:12 +00:00
|
|
|
module->astExpectedTypes[expr] = *expectedType;
|
2022-06-17 01:54:42 +01:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprConstantString* string, std::optional<TypeId> expectedType, bool forceSingleton)
|
2022-10-27 23:22:49 +01:00
|
|
|
{
|
2022-11-04 17:02:37 +00:00
|
|
|
if (forceSingleton)
|
2023-01-03 17:33:19 +00:00
|
|
|
return Inference{arena->addType(SingletonType{StringSingleton{std::string{string->value.data, string->value.size}}})};
|
2022-11-04 17:02:37 +00:00
|
|
|
|
2024-01-27 02:30:40 +00:00
|
|
|
FreeType ft = FreeType{scope.get()};
|
|
|
|
ft.lowerBound = arena->addType(SingletonType{StringSingleton{std::string{string->value.data, string->value.size}}});
|
|
|
|
ft.upperBound = builtinTypes->stringType;
|
|
|
|
const TypeId freeTy = arena->addType(ft);
|
|
|
|
addConstraint(scope, string->location, PrimitiveTypeConstraint{freeTy, expectedType, builtinTypes->stringType});
|
|
|
|
return Inference{freeTy};
|
2022-10-27 23:22:49 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprConstantBool* boolExpr, std::optional<TypeId> expectedType, bool forceSingleton)
|
2022-10-27 23:22:49 +01:00
|
|
|
{
|
2023-01-03 17:33:19 +00:00
|
|
|
const TypeId singletonType = boolExpr->value ? builtinTypes->trueType : builtinTypes->falseType;
|
2022-11-04 17:02:37 +00:00
|
|
|
if (forceSingleton)
|
|
|
|
return Inference{singletonType};
|
|
|
|
|
2024-01-27 02:30:40 +00:00
|
|
|
FreeType ft = FreeType{scope.get()};
|
|
|
|
ft.lowerBound = singletonType;
|
|
|
|
ft.upperBound = builtinTypes->booleanType;
|
|
|
|
const TypeId freeTy = arena->addType(ft);
|
|
|
|
addConstraint(scope, boolExpr->location, PrimitiveTypeConstraint{freeTy, expectedType, builtinTypes->booleanType});
|
|
|
|
return Inference{freeTy};
|
2022-10-27 23:22:49 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprLocal* local)
|
2022-10-21 18:33:43 +01:00
|
|
|
{
|
2023-10-20 21:36:26 +01:00
|
|
|
const RefinementKey* key = dfg->getRefinementKey(local);
|
|
|
|
std::optional<DefId> rvalueDef = dfg->getRValueDefForCompoundAssign(local);
|
|
|
|
LUAU_ASSERT(key || rvalueDef);
|
|
|
|
|
|
|
|
std::optional<TypeId> maybeTy;
|
|
|
|
|
|
|
|
// if we have a refinement key, we can look up its type.
|
|
|
|
if (key)
|
2024-01-27 02:30:40 +00:00
|
|
|
maybeTy = lookup(scope, key->def);
|
2023-10-20 21:36:26 +01:00
|
|
|
|
|
|
|
// 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)
|
2024-01-27 02:30:40 +00:00
|
|
|
maybeTy = lookup(scope, *rvalueDef);
|
2023-10-20 21:36:26 +01:00
|
|
|
|
|
|
|
if (maybeTy)
|
|
|
|
{
|
|
|
|
TypeId ty = follow(*maybeTy);
|
2023-11-17 18:15:31 +00:00
|
|
|
|
|
|
|
recordInferredBinding(local->local, ty);
|
2022-10-21 18:33:43 +01:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
return Inference{ty, refinementArena.proposition(key, builtinTypes->truthyType)};
|
|
|
|
}
|
2022-11-04 17:02:37 +00:00
|
|
|
else
|
2023-11-03 19:47:28 +00:00
|
|
|
ice->ice("CG: AstExprLocal came before its declaration?");
|
2022-10-21 18:33:43 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprGlobal* global)
|
2022-10-21 18:33:43 +01:00
|
|
|
{
|
2023-10-20 21:36:26 +01:00
|
|
|
const RefinementKey* key = dfg->getRefinementKey(global);
|
|
|
|
std::optional<DefId> 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;
|
2022-10-21 18:33:43 +01:00
|
|
|
|
|
|
|
/* 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.
|
|
|
|
*/
|
2024-01-27 02:30:40 +00:00
|
|
|
if (auto ty = lookup(scope, def, /*prototype=*/false))
|
2023-03-03 13:45:38 +00:00
|
|
|
{
|
2023-11-10 18:05:48 +00:00
|
|
|
rootScope->lvalueTypes[def] = *ty;
|
2023-10-20 21:36:26 +01:00
|
|
|
return Inference{*ty, refinementArena.proposition(key, builtinTypes->truthyType)};
|
2023-03-03 13:45:38 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-12-08 15:42:54 +00:00
|
|
|
reportError(global->location, UnknownSymbol{global->name.value, UnknownSymbol::Binding});
|
2023-03-03 13:45:38 +00:00
|
|
|
return Inference{builtinTypes->errorRecoveryType()};
|
|
|
|
}
|
2022-10-21 18:33:43 +01:00
|
|
|
}
|
|
|
|
|
2023-12-02 02:04:44 +00:00
|
|
|
Inference ConstraintGenerator::checkIndexName(const ScopePtr& scope, const RefinementKey* key, AstExpr* indexee, std::string index)
|
2022-06-17 01:54:42 +01:00
|
|
|
{
|
2023-12-02 02:04:44 +00:00
|
|
|
TypeId obj = check(scope, indexee).ty;
|
2023-02-24 18:24:22 +00:00
|
|
|
TypeId result = arena->addType(BlockedType{});
|
2022-12-02 10:46:05 +00:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
if (key)
|
2022-12-02 10:46:05 +00:00
|
|
|
{
|
2024-01-27 02:30:40 +00:00
|
|
|
if (auto ty = lookup(scope, key->def))
|
2023-10-20 21:36:26 +01:00
|
|
|
return Inference{*ty, refinementArena.proposition(key, builtinTypes->truthyType)};
|
2023-03-03 13:45:38 +00:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
scope->rvalueRefinements[key->def] = result;
|
2022-12-02 10:46:05 +00:00
|
|
|
}
|
2022-06-17 01:54:42 +01:00
|
|
|
|
2024-02-23 18:40:00 +00:00
|
|
|
addConstraint(scope, indexee->location, HasPropConstraint{result, obj, std::move(index), ValueContext::RValue});
|
2022-06-17 01:54:42 +01:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
if (key)
|
|
|
|
return Inference{result, refinementArena.proposition(key, builtinTypes->truthyType)};
|
2022-12-02 10:46:05 +00:00
|
|
|
else
|
|
|
|
return Inference{result};
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
|
2023-12-02 02:04:44 +00:00
|
|
|
Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprIndexName* indexName)
|
|
|
|
{
|
|
|
|
const RefinementKey* key = dfg->getRefinementKey(indexName);
|
|
|
|
return checkIndexName(scope, key, indexName->expr, indexName->index.value);
|
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprIndexExpr* indexExpr)
|
2022-07-01 00:29:02 +01:00
|
|
|
{
|
2023-12-02 02:04:44 +00:00
|
|
|
if (auto constantString = indexExpr->index->as<AstExprConstantString>())
|
|
|
|
{
|
|
|
|
const RefinementKey* key = dfg->getRefinementKey(indexExpr);
|
|
|
|
return checkIndexName(scope, key, indexExpr->expr, constantString->value.data);
|
|
|
|
}
|
|
|
|
|
2022-10-27 23:22:49 +01:00
|
|
|
TypeId obj = check(scope, indexExpr->expr).ty;
|
|
|
|
TypeId indexType = check(scope, indexExpr->index).ty;
|
2023-12-02 02:04:44 +00:00
|
|
|
|
2022-07-01 00:29:02 +01:00
|
|
|
TypeId result = freshType(scope);
|
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
const RefinementKey* key = dfg->getRefinementKey(indexExpr);
|
|
|
|
if (key)
|
2023-03-03 13:45:38 +00:00
|
|
|
{
|
2024-01-27 02:30:40 +00:00
|
|
|
if (auto ty = lookup(scope, key->def))
|
2023-10-20 21:36:26 +01:00
|
|
|
return Inference{*ty, refinementArena.proposition(key, builtinTypes->truthyType)};
|
2023-03-03 13:45:38 +00:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
scope->rvalueRefinements[key->def] = result;
|
2023-03-03 13:45:38 +00:00
|
|
|
}
|
|
|
|
|
2022-07-01 00:29:02 +01:00
|
|
|
TableIndexer indexer{indexType, result};
|
2023-01-03 17:33:19 +00:00
|
|
|
TypeId tableType = arena->addType(TableType{TableType::Props{}, TableIndexer{indexType, result}, TypeLevel{}, scope.get(), TableState::Free});
|
2022-07-01 00:29:02 +01:00
|
|
|
|
2022-09-02 00:00:14 +01:00
|
|
|
addConstraint(scope, indexExpr->expr->location, SubtypeConstraint{obj, tableType});
|
2022-07-01 00:29:02 +01:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
if (key)
|
|
|
|
return Inference{result, refinementArena.proposition(key, builtinTypes->truthyType)};
|
2023-03-03 13:45:38 +00:00
|
|
|
else
|
|
|
|
return Inference{result};
|
2022-07-01 00:29:02 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprFunction* func, std::optional<TypeId> expectedType, bool generalize)
|
2023-05-19 19:59:59 +01:00
|
|
|
{
|
|
|
|
Checkpoint startCheckpoint = checkpoint(this);
|
|
|
|
FunctionSignature sig = checkFunctionSignature(scope, func, expectedType);
|
2024-02-23 18:40:00 +00:00
|
|
|
|
|
|
|
interiorTypes.push_back(std::vector<TypeId>{});
|
2023-05-19 19:59:59 +01:00
|
|
|
checkFunctionBody(sig.bodyScope, func);
|
|
|
|
Checkpoint endCheckpoint = checkpoint(this);
|
|
|
|
|
2024-02-23 18:40:00 +00:00
|
|
|
TypeId generalizedTy = arena->addType(BlockedType{});
|
|
|
|
NotNull<Constraint> gc = addConstraint(sig.signatureScope, func->location, GeneralizationConstraint{generalizedTy, sig.signature, std::move(interiorTypes.back())});
|
|
|
|
interiorTypes.pop_back();
|
2023-05-19 19:59:59 +01:00
|
|
|
|
2024-02-23 18:40:00 +00:00
|
|
|
Constraint* previous = nullptr;
|
|
|
|
forEachConstraint(startCheckpoint, endCheckpoint, this, [gc, &previous](const ConstraintPtr& constraint) {
|
|
|
|
gc->dependencies.emplace_back(constraint.get());
|
2023-07-07 18:14:35 +01:00
|
|
|
|
2024-02-23 18:40:00 +00:00
|
|
|
if (auto psc = get<PackSubtypeConstraint>(*constraint); psc && psc->returns)
|
|
|
|
{
|
|
|
|
if (previous)
|
|
|
|
constraint->dependencies.push_back(NotNull{previous});
|
2023-07-07 18:14:35 +01:00
|
|
|
|
2024-02-23 18:40:00 +00:00
|
|
|
previous = constraint.get();
|
|
|
|
}
|
|
|
|
});
|
2023-05-19 19:59:59 +01:00
|
|
|
|
2024-02-23 18:40:00 +00:00
|
|
|
if (generalize && hasFreeType(sig.signature))
|
|
|
|
{
|
2023-09-30 01:22:06 +01:00
|
|
|
return Inference{generalizedTy};
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return Inference{sig.signature};
|
|
|
|
}
|
2023-05-19 19:59:59 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprUnary* unary)
|
2022-07-01 00:29:02 +01:00
|
|
|
{
|
2023-02-03 12:34:12 +00:00
|
|
|
auto [operandType, refinement] = check(scope, unary->expr);
|
2022-11-04 17:02:37 +00:00
|
|
|
|
2023-10-13 20:38:31 +01:00
|
|
|
switch (unary->op)
|
|
|
|
{
|
|
|
|
case AstExprUnary::Op::Not:
|
|
|
|
{
|
|
|
|
TypeId resultType = arena->addType(TypeFamilyInstanceType{
|
|
|
|
NotNull{&kBuiltinTypeFamilies.notFamily},
|
|
|
|
{operandType},
|
|
|
|
{},
|
|
|
|
});
|
|
|
|
addConstraint(scope, unary->location, ReduceConstraint{resultType});
|
2023-02-03 12:34:12 +00:00
|
|
|
return Inference{resultType, refinementArena.negation(refinement)};
|
2023-10-13 20:38:31 +01:00
|
|
|
}
|
2023-10-20 21:36:26 +01:00
|
|
|
case AstExprUnary::Op::Len:
|
2023-10-13 20:38:31 +01:00
|
|
|
{
|
2023-10-20 21:36:26 +01:00
|
|
|
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)};
|
2023-10-13 20:38:31 +01:00
|
|
|
}
|
2023-10-20 21:36:26 +01:00
|
|
|
default: // msvc can't prove that this is exhaustive.
|
|
|
|
LUAU_UNREACHABLE();
|
2023-10-13 20:38:31 +01:00
|
|
|
}
|
2022-10-21 18:33:43 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprBinary* binary, std::optional<TypeId> expectedType)
|
2022-10-21 18:33:43 +01:00
|
|
|
{
|
2023-02-03 12:34:12 +00:00
|
|
|
auto [leftType, rightType, refinement] = checkBinary(scope, binary, expectedType);
|
2022-10-21 18:33:43 +01:00
|
|
|
|
2023-10-06 18:31:16 +01:00
|
|
|
switch (binary->op)
|
|
|
|
{
|
|
|
|
case AstExprBinary::Op::Add:
|
2023-05-19 19:59:59 +01:00
|
|
|
{
|
|
|
|
TypeId resultType = arena->addType(TypeFamilyInstanceType{
|
|
|
|
NotNull{&kBuiltinTypeFamilies.addFamily},
|
|
|
|
{leftType, rightType},
|
|
|
|
{},
|
|
|
|
});
|
|
|
|
addConstraint(scope, binary->location, ReduceConstraint{resultType});
|
|
|
|
return Inference{resultType, std::move(refinement)};
|
|
|
|
}
|
2023-10-06 18:31:16 +01:00
|
|
|
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)};
|
|
|
|
}
|
2023-10-13 20:38:31 +01:00
|
|
|
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)};
|
|
|
|
}
|
2023-10-06 18:31:16 +01:00
|
|
|
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)};
|
|
|
|
}
|
2023-10-13 20:38:31 +01:00
|
|
|
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)};
|
|
|
|
}
|
2023-10-20 21:36:26 +01:00
|
|
|
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();
|
2023-10-06 18:31:16 +01:00
|
|
|
}
|
2022-07-01 00:29:02 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprIfElse* ifElse, std::optional<TypeId> expectedType)
|
2022-08-18 22:04:33 +01:00
|
|
|
{
|
2022-12-09 18:07:25 +00:00
|
|
|
ScopePtr condScope = childScope(ifElse->condition, scope);
|
2023-03-03 13:45:38 +00:00
|
|
|
RefinementId refinement = check(condScope, ifElse->condition).refinement;
|
2022-12-09 18:07:25 +00:00
|
|
|
|
|
|
|
ScopePtr thenScope = childScope(ifElse->trueExpr, scope);
|
2023-02-03 12:34:12 +00:00
|
|
|
applyRefinements(thenScope, ifElse->trueExpr->location, refinement);
|
2023-10-06 18:31:16 +01:00
|
|
|
TypeId thenType = check(thenScope, ifElse->trueExpr, expectedType).ty;
|
2022-08-18 22:04:33 +01:00
|
|
|
|
2022-12-09 18:07:25 +00:00
|
|
|
ScopePtr elseScope = childScope(ifElse->falseExpr, scope);
|
2023-02-03 12:34:12 +00:00
|
|
|
applyRefinements(elseScope, ifElse->falseExpr->location, refinementArena.negation(refinement));
|
2023-10-06 18:31:16 +01:00
|
|
|
TypeId elseType = check(elseScope, ifElse->falseExpr, expectedType).ty;
|
2022-08-18 22:04:33 +01:00
|
|
|
|
2024-01-27 02:30:40 +00:00
|
|
|
return Inference{expectedType ? *expectedType : makeUnion(scope, ifElse->location, thenType, elseType)};
|
2022-08-18 22:04:33 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprTypeAssertion* typeAssert)
|
2022-08-11 21:42:54 +01:00
|
|
|
{
|
2023-10-06 18:31:16 +01:00
|
|
|
check(scope, typeAssert->expr, std::nullopt);
|
2023-01-20 12:02:39 +00:00
|
|
|
return Inference{resolveType(scope, typeAssert->annotation, /* inTypeArguments */ false)};
|
2022-08-11 21:42:54 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprInterpString* interpString)
|
2023-01-27 21:28:45 +00:00
|
|
|
{
|
|
|
|
for (AstExpr* expr : interpString->expressions)
|
|
|
|
check(scope, expr);
|
|
|
|
|
|
|
|
return Inference{builtinTypes->stringType};
|
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
std::tuple<TypeId, TypeId, RefinementId> ConstraintGenerator::checkBinary(
|
2022-11-04 17:02:37 +00:00
|
|
|
const ScopePtr& scope, AstExprBinary* binary, std::optional<TypeId> expectedType)
|
|
|
|
{
|
|
|
|
if (binary->op == AstExprBinary::And)
|
|
|
|
{
|
2023-04-28 12:55:55 +01:00
|
|
|
std::optional<TypeId> relaxedExpectedLhs;
|
|
|
|
|
|
|
|
if (expectedType)
|
|
|
|
relaxedExpectedLhs = arena->addType(UnionType{{builtinTypes->falsyType, *expectedType}});
|
|
|
|
|
2023-10-06 18:31:16 +01:00
|
|
|
auto [leftType, leftRefinement] = check(scope, binary->left, relaxedExpectedLhs);
|
2022-11-04 17:02:37 +00:00
|
|
|
|
|
|
|
ScopePtr rightScope = childScope(binary->right, scope);
|
2023-02-03 12:34:12 +00:00
|
|
|
applyRefinements(rightScope, binary->right->location, leftRefinement);
|
2023-10-06 18:31:16 +01:00
|
|
|
auto [rightType, rightRefinement] = check(rightScope, binary->right, expectedType);
|
2022-11-04 17:02:37 +00:00
|
|
|
|
2023-02-03 12:34:12 +00:00
|
|
|
return {leftType, rightType, refinementArena.conjunction(leftRefinement, rightRefinement)};
|
2022-11-04 17:02:37 +00:00
|
|
|
}
|
|
|
|
else if (binary->op == AstExprBinary::Or)
|
|
|
|
{
|
2023-04-28 12:55:55 +01:00
|
|
|
std::optional<TypeId> relaxedExpectedLhs;
|
|
|
|
|
|
|
|
if (expectedType)
|
|
|
|
relaxedExpectedLhs = arena->addType(UnionType{{builtinTypes->falsyType, *expectedType}});
|
|
|
|
|
2023-10-06 18:31:16 +01:00
|
|
|
auto [leftType, leftRefinement] = check(scope, binary->left, relaxedExpectedLhs);
|
2022-11-04 17:02:37 +00:00
|
|
|
|
|
|
|
ScopePtr rightScope = childScope(binary->right, scope);
|
2023-02-03 12:34:12 +00:00
|
|
|
applyRefinements(rightScope, binary->right->location, refinementArena.negation(leftRefinement));
|
2023-10-06 18:31:16 +01:00
|
|
|
auto [rightType, rightRefinement] = check(rightScope, binary->right, expectedType);
|
2022-11-04 17:02:37 +00:00
|
|
|
|
2023-02-03 12:34:12 +00:00
|
|
|
return {leftType, rightType, refinementArena.disjunction(leftRefinement, rightRefinement)};
|
2022-11-04 17:02:37 +00:00
|
|
|
}
|
2022-11-10 22:04:44 +00:00
|
|
|
else if (auto typeguard = matchTypeGuard(binary))
|
|
|
|
{
|
|
|
|
TypeId leftType = check(scope, binary->left).ty;
|
|
|
|
TypeId rightType = check(scope, binary->right).ty;
|
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
const RefinementKey* key = dfg->getRefinementKey(typeguard->target);
|
|
|
|
if (!key)
|
2022-11-10 22:04:44 +00:00
|
|
|
return {leftType, rightType, nullptr};
|
2023-10-20 21:36:26 +01:00
|
|
|
|
2023-07-28 12:37:00 +01:00
|
|
|
auto augmentForErrorSupression = [&](TypeId ty) -> TypeId {
|
|
|
|
return arena->addType(UnionType{{ty, builtinTypes->errorType}});
|
|
|
|
};
|
2022-11-10 22:04:44 +00:00
|
|
|
|
2023-01-03 17:33:19 +00:00
|
|
|
TypeId discriminantTy = builtinTypes->neverType;
|
2022-11-10 22:04:44 +00:00
|
|
|
if (typeguard->type == "nil")
|
2023-01-03 17:33:19 +00:00
|
|
|
discriminantTy = builtinTypes->nilType;
|
2022-11-10 22:04:44 +00:00
|
|
|
else if (typeguard->type == "string")
|
2023-01-03 17:33:19 +00:00
|
|
|
discriminantTy = builtinTypes->stringType;
|
2022-11-10 22:04:44 +00:00
|
|
|
else if (typeguard->type == "number")
|
2023-01-03 17:33:19 +00:00
|
|
|
discriminantTy = builtinTypes->numberType;
|
2022-11-10 22:04:44 +00:00
|
|
|
else if (typeguard->type == "boolean")
|
2023-02-24 18:24:22 +00:00
|
|
|
discriminantTy = builtinTypes->booleanType;
|
|
|
|
else if (typeguard->type == "thread")
|
2023-01-03 17:33:19 +00:00
|
|
|
discriminantTy = builtinTypes->threadType;
|
2023-11-10 18:05:48 +00:00
|
|
|
else if (typeguard->type == "buffer")
|
|
|
|
discriminantTy = builtinTypes->bufferType;
|
2022-11-10 22:04:44 +00:00
|
|
|
else if (typeguard->type == "table")
|
2023-07-28 12:37:00 +01:00
|
|
|
discriminantTy = augmentForErrorSupression(builtinTypes->tableType);
|
2022-11-10 22:04:44 +00:00
|
|
|
else if (typeguard->type == "function")
|
2023-07-28 12:37:00 +01:00
|
|
|
discriminantTy = augmentForErrorSupression(builtinTypes->functionType);
|
2022-11-10 22:04:44 +00:00
|
|
|
else if (typeguard->type == "userdata")
|
|
|
|
{
|
2023-02-24 18:24:22 +00:00
|
|
|
// For now, we don't really care about being accurate with userdata if the typeguard was using typeof.
|
|
|
|
discriminantTy = builtinTypes->classType;
|
2022-11-10 22:04:44 +00:00
|
|
|
}
|
|
|
|
else if (!typeguard->isTypeof && typeguard->type == "vector")
|
2023-01-03 17:33:19 +00:00
|
|
|
discriminantTy = builtinTypes->neverType; // TODO: figure out a way to deal with this quirky type
|
2022-11-10 22:04:44 +00:00
|
|
|
else if (!typeguard->isTypeof)
|
2023-01-03 17:33:19 +00:00
|
|
|
discriminantTy = builtinTypes->neverType;
|
2022-11-10 22:04:44 +00:00
|
|
|
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.
|
2023-05-05 20:57:12 +01:00
|
|
|
if (auto ctv = get<ClassType>(ty); !ctv || ctv->parent == builtinTypes->classType)
|
2022-11-10 22:04:44 +00:00
|
|
|
discriminantTy = ty;
|
|
|
|
}
|
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
RefinementId proposition = refinementArena.proposition(key, discriminantTy);
|
2022-11-10 22:04:44 +00:00
|
|
|
if (binary->op == AstExprBinary::CompareEq)
|
|
|
|
return {leftType, rightType, proposition};
|
|
|
|
else if (binary->op == AstExprBinary::CompareNe)
|
2023-02-03 12:34:12 +00:00
|
|
|
return {leftType, rightType, refinementArena.negation(proposition)};
|
2022-11-10 22:04:44 +00:00
|
|
|
else
|
|
|
|
ice->ice("matchTypeGuard should only return a Some under `==` or `~=`!");
|
|
|
|
}
|
2022-11-04 17:02:37 +00:00
|
|
|
else if (binary->op == AstExprBinary::CompareEq || binary->op == AstExprBinary::CompareNe)
|
|
|
|
{
|
2023-05-05 20:57:12 +01:00
|
|
|
// 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
|
2023-10-06 18:31:16 +01:00
|
|
|
TypeId leftType = check(scope, binary->left, {}, true).ty;
|
|
|
|
TypeId rightType = check(scope, binary->right, {}, true).ty;
|
2022-11-04 17:02:37 +00:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
RefinementId leftRefinement = refinementArena.proposition(dfg->getRefinementKey(binary->left), rightType);
|
|
|
|
RefinementId rightRefinement = refinementArena.proposition(dfg->getRefinementKey(binary->right), leftType);
|
2022-11-04 17:02:37 +00:00
|
|
|
|
|
|
|
if (binary->op == AstExprBinary::CompareNe)
|
|
|
|
{
|
2023-02-03 12:34:12 +00:00
|
|
|
leftRefinement = refinementArena.negation(leftRefinement);
|
|
|
|
rightRefinement = refinementArena.negation(rightRefinement);
|
2022-11-04 17:02:37 +00:00
|
|
|
}
|
|
|
|
|
2023-02-03 12:34:12 +00:00
|
|
|
return {leftType, rightType, refinementArena.equivalence(leftRefinement, rightRefinement)};
|
2022-11-04 17:02:37 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-10-06 18:31:16 +01:00
|
|
|
TypeId leftType = check(scope, binary->left).ty;
|
|
|
|
TypeId rightType = check(scope, binary->right).ty;
|
2022-11-04 17:02:37 +00:00
|
|
|
return {leftType, rightType, nullptr};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-19 15:13:08 +00:00
|
|
|
std::optional<TypeId> ConstraintGenerator::checkLValue(const ScopePtr& scope, AstExpr* expr, TypeId assignedTy, bool transform)
|
2023-10-06 18:31:16 +01:00
|
|
|
{
|
|
|
|
if (auto local = expr->as<AstExprLocal>())
|
2024-01-19 15:13:08 +00:00
|
|
|
return checkLValue(scope, local, assignedTy, transform);
|
2023-10-06 18:31:16 +01:00
|
|
|
else if (auto global = expr->as<AstExprGlobal>())
|
2023-11-03 19:47:28 +00:00
|
|
|
return checkLValue(scope, global, assignedTy);
|
2023-10-06 18:31:16 +01:00
|
|
|
else if (auto indexName = expr->as<AstExprIndexName>())
|
2023-11-03 19:47:28 +00:00
|
|
|
return checkLValue(scope, indexName, assignedTy);
|
2023-10-06 18:31:16 +01:00
|
|
|
else if (auto indexExpr = expr->as<AstExprIndexExpr>())
|
2023-11-03 19:47:28 +00:00
|
|
|
return checkLValue(scope, indexExpr, assignedTy);
|
2023-10-06 18:31:16 +01:00
|
|
|
else if (auto error = expr->as<AstExprError>())
|
|
|
|
{
|
|
|
|
check(scope, error);
|
|
|
|
return builtinTypes->errorRecoveryType();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
ice->ice("checkLValue is inexhaustive");
|
|
|
|
}
|
|
|
|
|
2024-01-19 15:13:08 +00:00
|
|
|
std::optional<TypeId> ConstraintGenerator::checkLValue(const ScopePtr& scope, AstExprLocal* local, TypeId assignedTy, bool transform)
|
2023-10-06 18:31:16 +01:00
|
|
|
{
|
2023-10-20 21:36:26 +01:00
|
|
|
std::optional<TypeId> annotatedTy = scope->lookup(local->local);
|
2023-11-17 18:15:31 +00:00
|
|
|
LUAU_ASSERT(annotatedTy);
|
2023-10-20 21:36:26 +01:00
|
|
|
if (annotatedTy)
|
2023-11-03 19:47:28 +00:00
|
|
|
addConstraint(scope, local->location, SubtypeConstraint{assignedTy, *annotatedTy});
|
2023-10-20 21:36:26 +01:00
|
|
|
|
2023-11-17 18:15:31 +00:00
|
|
|
const DefId defId = dfg->getDef(local);
|
|
|
|
std::optional<TypeId> ty = scope->lookupUnrefinedType(defId);
|
|
|
|
|
|
|
|
if (ty)
|
|
|
|
{
|
2024-01-19 15:13:08 +00:00
|
|
|
if (transform)
|
|
|
|
{
|
|
|
|
if (auto lt = getMutable<LocalType>(*ty))
|
|
|
|
++lt->blockCount;
|
2024-02-02 18:20:03 +00:00
|
|
|
else if (auto ut = getMutable<UnionType>(*ty))
|
|
|
|
{
|
|
|
|
for (TypeId optTy : ut->options)
|
|
|
|
if (auto lt = getMutable<LocalType>(optTy))
|
|
|
|
++lt->blockCount;
|
|
|
|
}
|
2024-01-19 15:13:08 +00:00
|
|
|
}
|
2023-11-17 18:15:31 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ty = arena->addType(LocalType{builtinTypes->neverType, /* blockCount */ 1, local->local->name.value});
|
|
|
|
|
2024-02-02 18:20:03 +00:00
|
|
|
if (annotatedTy)
|
|
|
|
{
|
|
|
|
switch (shouldSuppressErrors(normalizer, *annotatedTy))
|
|
|
|
{
|
|
|
|
case ErrorSuppression::DoNotSuppress:
|
|
|
|
break;
|
|
|
|
case ErrorSuppression::Suppress:
|
|
|
|
ty = simplifyUnion(builtinTypes, arena, *ty, builtinTypes->errorType).result;
|
|
|
|
break;
|
|
|
|
case ErrorSuppression::NormalizationFailed:
|
|
|
|
reportError(local->local->annotation->location, NormalizationTooComplex{});
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-17 18:15:31 +00:00
|
|
|
scope->lvalueTypes[defId] = *ty;
|
|
|
|
}
|
|
|
|
|
2024-01-19 15:13:08 +00:00
|
|
|
if (transform)
|
|
|
|
{
|
2024-02-16 01:25:31 +00:00
|
|
|
addConstraint(scope, local->location,
|
|
|
|
UnpackConstraint{arena->addTypePack({*ty}), arena->addTypePack({assignedTy}),
|
|
|
|
/*resultIsLValue*/ true});
|
2024-01-19 15:13:08 +00:00
|
|
|
|
|
|
|
recordInferredBinding(local->local, *ty);
|
|
|
|
}
|
2023-11-17 18:15:31 +00:00
|
|
|
|
|
|
|
|
|
|
|
return ty;
|
2023-10-06 18:31:16 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
std::optional<TypeId> ConstraintGenerator::checkLValue(const ScopePtr& scope, AstExprGlobal* global, TypeId assignedTy)
|
2023-10-06 18:31:16 +01:00
|
|
|
{
|
2023-10-20 21:36:26 +01:00
|
|
|
return scope->lookup(Symbol{global->name});
|
2023-10-06 18:31:16 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
std::optional<TypeId> ConstraintGenerator::checkLValue(const ScopePtr& scope, AstExprIndexName* indexName, TypeId assignedTy)
|
2023-10-06 18:31:16 +01:00
|
|
|
{
|
2023-11-03 19:47:28 +00:00
|
|
|
return updateProperty(scope, indexName, assignedTy);
|
2023-10-06 18:31:16 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
std::optional<TypeId> ConstraintGenerator::checkLValue(const ScopePtr& scope, AstExprIndexExpr* indexExpr, TypeId assignedTy)
|
2023-10-06 18:31:16 +01:00
|
|
|
{
|
2023-11-03 19:47:28 +00:00
|
|
|
return updateProperty(scope, indexExpr, assignedTy);
|
2023-03-03 13:45:38 +00:00
|
|
|
}
|
|
|
|
|
2022-10-21 18:33:43 +01:00
|
|
|
/**
|
|
|
|
* This function is mostly about identifying properties that are being inserted into unsealed tables.
|
|
|
|
*
|
|
|
|
* If expr has the form name.a.b.c
|
|
|
|
*/
|
2023-11-03 19:47:28 +00:00
|
|
|
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<AstExprIndexName>() || expr->is<AstExprIndexExpr>());
|
|
|
|
|
2023-03-03 13:45:38 +00:00
|
|
|
if (auto indexExpr = expr->as<AstExprIndexExpr>(); indexExpr && !indexExpr->index->is<AstExprConstantString>())
|
2022-10-21 18:33:43 +01:00
|
|
|
{
|
2023-02-24 18:24:22 +00:00
|
|
|
// 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;
|
2023-11-03 19:47:28 +00:00
|
|
|
addConstraint(scope, expr->location, SetIndexerConstraint{resultType, subjectType, indexType, assignedTy});
|
2023-02-24 18:24:22 +00:00
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
module->astTypes[expr] = assignedTy;
|
2023-02-24 18:24:22 +00:00
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
return assignedTy;
|
2022-10-21 18:33:43 +01:00
|
|
|
}
|
|
|
|
|
2023-01-20 12:02:39 +00:00
|
|
|
Symbol sym;
|
2023-10-20 21:36:26 +01:00
|
|
|
const Def* def = nullptr;
|
2023-01-20 12:02:39 +00:00
|
|
|
std::vector<std::string> segments;
|
|
|
|
std::vector<AstExpr*> exprs;
|
|
|
|
|
|
|
|
AstExpr* e = expr;
|
|
|
|
while (e)
|
|
|
|
{
|
|
|
|
if (auto global = e->as<AstExprGlobal>())
|
|
|
|
{
|
|
|
|
sym = global->name;
|
2023-10-20 21:36:26 +01:00
|
|
|
def = dfg->getDef(global);
|
2023-01-20 12:02:39 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
else if (auto local = e->as<AstExprLocal>())
|
|
|
|
{
|
|
|
|
sym = local->local;
|
2023-10-20 21:36:26 +01:00
|
|
|
def = dfg->getDef(local);
|
2023-01-20 12:02:39 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
else if (auto indexName = e->as<AstExprIndexName>())
|
|
|
|
{
|
|
|
|
segments.push_back(indexName->index.value);
|
|
|
|
exprs.push_back(e);
|
|
|
|
e = indexName->expr;
|
|
|
|
}
|
2023-03-03 13:45:38 +00:00
|
|
|
else if (auto indexExpr = e->as<AstExprIndexExpr>())
|
|
|
|
{
|
|
|
|
if (auto strIndex = indexExpr->index->as<AstExprConstantString>())
|
|
|
|
{
|
2023-11-03 19:47:28 +00:00
|
|
|
// We need to populate astTypes for the index value.
|
|
|
|
check(scope, indexExpr->index);
|
|
|
|
|
2023-03-03 13:45:38 +00:00
|
|
|
segments.push_back(std::string(strIndex->value.data, strIndex->value.size));
|
|
|
|
exprs.push_back(e);
|
|
|
|
e = indexExpr->expr;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-11-03 19:47:28 +00:00
|
|
|
return fallback();
|
2023-03-03 13:45:38 +00:00
|
|
|
}
|
|
|
|
}
|
2023-01-20 12:02:39 +00:00
|
|
|
else
|
2023-11-03 19:47:28 +00:00
|
|
|
{
|
|
|
|
return fallback();
|
|
|
|
}
|
2023-01-20 12:02:39 +00:00
|
|
|
}
|
2022-10-21 18:33:43 +01:00
|
|
|
|
2022-11-18 18:45:14 +00:00
|
|
|
LUAU_ASSERT(!segments.empty());
|
2022-10-21 18:33:43 +01:00
|
|
|
|
2023-01-20 12:02:39 +00:00
|
|
|
std::reverse(begin(segments), end(segments));
|
|
|
|
std::reverse(begin(exprs), end(exprs));
|
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
LUAU_ASSERT(def);
|
|
|
|
std::optional<std::pair<TypeId, Scope*>> lookupResult = scope->lookupEx(NotNull{def});
|
2022-10-21 18:33:43 +01:00
|
|
|
if (!lookupResult)
|
2023-11-03 19:47:28 +00:00
|
|
|
return fallback();
|
2023-10-13 20:38:31 +01:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
const auto [subjectType, subjectScope] = *lookupResult;
|
2022-10-21 18:33:43 +01:00
|
|
|
|
2022-11-18 18:45:14 +00:00
|
|
|
std::vector<std::string> segmentStrings(begin(segments), end(segments));
|
|
|
|
|
2023-01-03 17:33:19 +00:00
|
|
|
TypeId updatedType = arena->addType(BlockedType{});
|
2023-11-03 19:47:28 +00:00
|
|
|
addConstraint(scope, expr->location, SetPropConstraint{updatedType, subjectType, std::move(segmentStrings), assignedTy});
|
2022-10-21 18:33:43 +01:00
|
|
|
|
2023-01-20 12:02:39 +00:00
|
|
|
TypeId prevSegmentTy = updatedType;
|
|
|
|
for (size_t i = 0; i < segments.size(); ++i)
|
|
|
|
{
|
|
|
|
TypeId segmentTy = arena->addType(BlockedType{});
|
2023-02-03 12:34:12 +00:00
|
|
|
module->astTypes[exprs[i]] = segmentTy;
|
2024-02-23 18:40:00 +00:00
|
|
|
ValueContext ctx = i == segments.size() - 1 ? ValueContext::LValue : ValueContext::RValue;
|
|
|
|
addConstraint(scope, expr->location, HasPropConstraint{segmentTy, prevSegmentTy, segments[i], ctx});
|
2023-01-20 12:02:39 +00:00
|
|
|
prevSegmentTy = segmentTy;
|
|
|
|
}
|
|
|
|
|
2023-02-03 12:34:12 +00:00
|
|
|
module->astTypes[expr] = prevSegmentTy;
|
|
|
|
module->astTypes[e] = updatedType;
|
2023-02-10 18:50:54 +00:00
|
|
|
|
2023-02-24 18:24:22 +00:00
|
|
|
if (!subjectType->persistent)
|
2023-02-10 18:50:54 +00:00
|
|
|
{
|
2023-10-20 21:36:26 +01:00
|
|
|
subjectScope->bindings[sym].typeId = updatedType;
|
2023-02-24 18:24:22 +00:00
|
|
|
|
2023-03-03 13:45:38 +00:00
|
|
|
// This can fail if the user is erroneously trying to augment a builtin
|
|
|
|
// table like os or string.
|
2023-10-20 21:36:26 +01:00
|
|
|
if (auto key = dfg->getRefinementKey(e))
|
2023-10-06 18:31:16 +01:00
|
|
|
{
|
2023-10-20 21:36:26 +01:00
|
|
|
subjectScope->lvalueTypes[key->def] = updatedType;
|
|
|
|
subjectScope->rvalueRefinements[key->def] = updatedType;
|
2023-10-06 18:31:16 +01:00
|
|
|
}
|
2023-02-10 18:50:54 +00:00
|
|
|
}
|
2022-11-18 18:45:14 +00:00
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
return assignedTy;
|
2022-10-21 18:33:43 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprTable* expr, std::optional<TypeId> expectedType)
|
2022-06-17 01:54:42 +01:00
|
|
|
{
|
2023-08-04 18:01:35 +01:00
|
|
|
const bool expectedTypeIsFree = expectedType && get<FreeType>(follow(*expectedType));
|
|
|
|
|
2023-01-03 17:33:19 +00:00
|
|
|
TypeId ty = arena->addType(TableType{});
|
|
|
|
TableType* ttv = getMutable<TableType>(ty);
|
2022-06-17 01:54:42 +01:00
|
|
|
LUAU_ASSERT(ttv);
|
|
|
|
|
2022-09-29 23:11:54 +01:00
|
|
|
ttv->state = TableState::Unsealed;
|
|
|
|
ttv->scope = scope.get();
|
|
|
|
|
2024-02-23 18:40:00 +00:00
|
|
|
interiorTypes.back().push_back(ty);
|
|
|
|
|
2022-09-02 00:00:14 +01:00
|
|
|
auto createIndexer = [this, scope, ttv](const Location& location, TypeId currentIndexType, TypeId currentResultType) {
|
2022-06-17 01:54:42 +01:00
|
|
|
if (!ttv->indexer)
|
2022-06-03 21:32:20 +01:00
|
|
|
{
|
2022-06-17 01:54:42 +01:00
|
|
|
TypeId indexType = this->freshType(scope);
|
|
|
|
TypeId resultType = this->freshType(scope);
|
|
|
|
ttv->indexer = TableIndexer{indexType, resultType};
|
2022-06-03 21:32:20 +01:00
|
|
|
}
|
|
|
|
|
2022-09-02 00:00:14 +01:00
|
|
|
addConstraint(scope, location, SubtypeConstraint{ttv->indexer->indexType, currentIndexType});
|
|
|
|
addConstraint(scope, location, SubtypeConstraint{ttv->indexer->indexResultType, currentResultType});
|
2022-06-17 01:54:42 +01:00
|
|
|
};
|
2022-06-03 21:32:20 +01:00
|
|
|
|
2023-02-10 18:50:54 +00:00
|
|
|
std::optional<TypeId> annotatedKeyType;
|
|
|
|
std::optional<TypeId> annotatedIndexResultType;
|
|
|
|
|
|
|
|
if (expectedType)
|
|
|
|
{
|
|
|
|
if (const TableType* ttv = get<TableType>(follow(*expectedType)))
|
|
|
|
{
|
|
|
|
if (ttv->indexer)
|
|
|
|
{
|
|
|
|
annotatedKeyType.emplace(follow(ttv->indexer->indexType));
|
|
|
|
annotatedIndexResultType.emplace(ttv->indexer->indexResultType);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isIndexedResultType = false;
|
|
|
|
std::optional<TypeId> pinnedIndexResultType;
|
|
|
|
|
|
|
|
|
2022-06-17 01:54:42 +01:00
|
|
|
for (const AstExprTable::Item& item : expr->items)
|
|
|
|
{
|
2022-09-23 19:32:10 +01:00
|
|
|
std::optional<TypeId> expectedValueType;
|
2023-02-10 18:50:54 +00:00
|
|
|
if (item.kind == AstExprTable::Item::Kind::General || item.kind == AstExprTable::Item::Kind::List)
|
|
|
|
isIndexedResultType = true;
|
2022-09-23 19:32:10 +01:00
|
|
|
|
2023-08-04 18:01:35 +01:00
|
|
|
if (item.key && expectedType && !expectedTypeIsFree)
|
2022-09-23 19:32:10 +01:00
|
|
|
{
|
|
|
|
if (auto stringKey = item.key->as<AstExprConstantString>())
|
|
|
|
{
|
2022-12-02 10:46:05 +00:00
|
|
|
ErrorVec errorVec;
|
|
|
|
std::optional<TypeId> propTy =
|
2023-01-03 17:33:19 +00:00
|
|
|
findTablePropertyRespectingMeta(builtinTypes, errorVec, follow(*expectedType), stringKey->value.data, item.value->location);
|
2022-12-02 10:46:05 +00:00
|
|
|
if (propTy)
|
|
|
|
expectedValueType = propTy;
|
|
|
|
else
|
|
|
|
{
|
2023-01-03 17:33:19 +00:00
|
|
|
expectedValueType = arena->addType(BlockedType{});
|
2023-05-19 19:59:59 +01:00
|
|
|
addConstraint(scope, item.value->location,
|
2024-02-23 18:40:00 +00:00
|
|
|
HasPropConstraint{
|
|
|
|
*expectedValueType, *expectedType, stringKey->value.data, ValueContext::RValue, /*suppressSimplification*/ true});
|
2022-12-02 10:46:05 +00:00
|
|
|
}
|
2022-09-23 19:32:10 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-10 18:50:54 +00:00
|
|
|
// 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<TypeId> checkExpectedIndexResultType;
|
|
|
|
if (expectedValueType)
|
|
|
|
{
|
|
|
|
checkExpectedIndexResultType = expectedValueType;
|
|
|
|
}
|
|
|
|
else if (annotatedIndexResultType)
|
|
|
|
{
|
|
|
|
checkExpectedIndexResultType = annotatedIndexResultType;
|
|
|
|
}
|
|
|
|
else if (pinnedIndexResultType)
|
|
|
|
{
|
|
|
|
checkExpectedIndexResultType = pinnedIndexResultType;
|
|
|
|
}
|
|
|
|
|
2023-10-06 18:31:16 +01:00
|
|
|
TypeId itemTy = check(scope, item.value, checkExpectedIndexResultType).ty;
|
2023-02-10 18:50:54 +00:00
|
|
|
|
2024-02-02 18:20:03 +00:00
|
|
|
// we should preserve error-suppressingness from the expected value type if we have one
|
|
|
|
if (expectedValueType)
|
|
|
|
{
|
|
|
|
switch (shouldSuppressErrors(normalizer, *expectedValueType))
|
|
|
|
{
|
|
|
|
case ErrorSuppression::DoNotSuppress:
|
|
|
|
break;
|
|
|
|
case ErrorSuppression::Suppress:
|
|
|
|
itemTy = simplifyUnion(builtinTypes, arena, itemTy, builtinTypes->errorType).result;
|
|
|
|
break;
|
|
|
|
case ErrorSuppression::NormalizationFailed:
|
|
|
|
reportError(item.value->location, NormalizationTooComplex{});
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-10 18:50:54 +00:00
|
|
|
if (isIndexedResultType && !pinnedIndexResultType)
|
|
|
|
pinnedIndexResultType = itemTy;
|
2022-06-03 21:32:20 +01:00
|
|
|
|
2022-06-17 01:54:42 +01:00
|
|
|
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.
|
2023-10-06 18:31:16 +01:00
|
|
|
TypeId keyTy = check(scope, item.key, annotatedKeyType).ty;
|
2022-06-17 01:54:42 +01:00
|
|
|
|
|
|
|
if (AstExprConstantString* key = item.key->as<AstExprConstantString>())
|
|
|
|
{
|
|
|
|
ttv->props[key->value.begin()] = {itemTy};
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-09-02 00:00:14 +01:00
|
|
|
createIndexer(item.key->location, keyTy, itemTy);
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-01-03 17:33:19 +00:00
|
|
|
TypeId numberType = builtinTypes->numberType;
|
2022-09-02 00:00:14 +01:00
|
|
|
// FIXME? The location isn't quite right here. Not sure what is
|
|
|
|
// right.
|
|
|
|
createIndexer(item.value->location, numberType, itemTy);
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
2022-06-03 21:32:20 +01:00
|
|
|
}
|
2022-06-17 01:54:42 +01:00
|
|
|
|
2022-10-27 23:22:49 +01:00
|
|
|
return Inference{ty};
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
ConstraintGenerator::FunctionSignature ConstraintGenerator::checkFunctionSignature(
|
2023-05-19 19:59:59 +01:00
|
|
|
const ScopePtr& parent, AstExprFunction* fn, std::optional<TypeId> expectedType, std::optional<Location> originalName)
|
2022-06-17 01:54:42 +01:00
|
|
|
{
|
2022-07-29 04:41:13 +01:00
|
|
|
ScopePtr signatureScope = nullptr;
|
|
|
|
ScopePtr bodyScope = nullptr;
|
2022-07-01 00:29:02 +01:00
|
|
|
TypePackId returnType = nullptr;
|
|
|
|
|
|
|
|
std::vector<TypeId> genericTypes;
|
|
|
|
std::vector<TypePackId> genericTypePacks;
|
|
|
|
|
2022-12-02 10:46:05 +00:00
|
|
|
if (expectedType)
|
|
|
|
expectedType = follow(*expectedType);
|
|
|
|
|
2022-07-01 00:29:02 +01:00
|
|
|
bool hasGenerics = fn->generics.size > 0 || fn->genericPacks.size > 0;
|
|
|
|
|
2022-12-02 10:46:05 +00:00
|
|
|
signatureScope = childScope(fn, parent);
|
2022-07-01 00:29:02 +01:00
|
|
|
|
2022-12-02 10:46:05 +00:00
|
|
|
// We need to assign returnType before creating bodyScope so that the
|
|
|
|
// return type gets propogated to bodyScope.
|
|
|
|
returnType = freshTypePack(signatureScope);
|
|
|
|
signatureScope->returnType = returnType;
|
2022-07-01 00:29:02 +01:00
|
|
|
|
2022-12-02 10:46:05 +00:00
|
|
|
bodyScope = childScope(fn->body, signatureScope);
|
2022-07-01 00:29:02 +01:00
|
|
|
|
2022-12-02 10:46:05 +00:00
|
|
|
if (hasGenerics)
|
|
|
|
{
|
2022-07-29 04:41:13 +01:00
|
|
|
std::vector<std::pair<Name, GenericTypeDefinition>> genericDefinitions = createGenerics(signatureScope, fn->generics);
|
|
|
|
std::vector<std::pair<Name, GenericTypePackDefinition>> genericPackDefinitions = createGenericPacks(signatureScope, fn->genericPacks);
|
2022-07-01 00:29:02 +01:00
|
|
|
|
|
|
|
// 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);
|
|
|
|
}
|
2022-12-02 10:46:05 +00:00
|
|
|
|
2022-12-09 18:07:25 +00:00
|
|
|
// Local variable works around an odd gcc 11.3 warning: <anonymous> may be used uninitialized
|
|
|
|
std::optional<TypeId> none = std::nullopt;
|
|
|
|
expectedType = none;
|
2022-07-01 00:29:02 +01:00
|
|
|
}
|
2022-12-02 10:46:05 +00:00
|
|
|
|
|
|
|
std::vector<TypeId> argTypes;
|
2023-02-17 14:53:37 +00:00
|
|
|
std::vector<std::optional<FunctionArgument>> argNames;
|
2022-12-02 10:46:05 +00:00
|
|
|
TypePack expectedArgPack;
|
|
|
|
|
2023-01-03 17:33:19 +00:00
|
|
|
const FunctionType* expectedFunction = expectedType ? get<FunctionType>(*expectedType) : nullptr;
|
2023-03-24 17:34:14 +00:00
|
|
|
// This check ensures that expectedType is precisely optional and not any (since any is also an optional type)
|
|
|
|
if (expectedType && isOptional(*expectedType) && !get<AnyType>(*expectedType))
|
|
|
|
{
|
|
|
|
auto ut = get<UnionType>(*expectedType);
|
|
|
|
for (auto u : ut)
|
|
|
|
{
|
|
|
|
if (get<FunctionType>(u) && !isNil(u))
|
|
|
|
{
|
|
|
|
expectedFunction = get<FunctionType>(u);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-12-02 10:46:05 +00:00
|
|
|
|
|
|
|
if (expectedFunction)
|
2022-07-01 00:29:02 +01:00
|
|
|
{
|
2023-01-03 17:33:19 +00:00
|
|
|
expectedArgPack = extendTypePack(*arena, builtinTypes, expectedFunction->argTypes, fn->args.size);
|
2022-12-02 10:46:05 +00:00
|
|
|
|
|
|
|
genericTypes = expectedFunction->generics;
|
|
|
|
genericTypePacks = expectedFunction->genericPacks;
|
|
|
|
}
|
2022-07-01 00:29:02 +01:00
|
|
|
|
2023-02-17 14:53:37 +00:00
|
|
|
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};
|
2023-03-03 13:45:38 +00:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
DefId def = dfg->getDef(fn->self);
|
|
|
|
signatureScope->lvalueTypes[def] = selfType;
|
|
|
|
signatureScope->rvalueRefinements[def] = selfType;
|
2023-02-17 14:53:37 +00:00
|
|
|
}
|
|
|
|
|
2022-12-02 10:46:05 +00:00
|
|
|
for (size_t i = 0; i < fn->args.size; ++i)
|
|
|
|
{
|
|
|
|
AstLocal* local = fn->args.data[i];
|
|
|
|
|
2023-03-03 13:45:38 +00:00
|
|
|
TypeId argTy = nullptr;
|
2022-12-02 10:46:05 +00:00
|
|
|
if (local->annotation)
|
2023-03-03 13:45:38 +00:00
|
|
|
argTy = resolveType(signatureScope, local->annotation, /* inTypeArguments */ false, /* replaceErrorWithFresh*/ true);
|
|
|
|
else
|
2022-12-02 10:46:05 +00:00
|
|
|
{
|
2023-03-03 13:45:38 +00:00
|
|
|
if (i < expectedArgPack.head.size())
|
2023-09-30 01:22:06 +01:00
|
|
|
argTy = expectedArgPack.head[i];
|
|
|
|
else
|
|
|
|
argTy = freshType(signatureScope);
|
2022-12-02 10:46:05 +00:00
|
|
|
}
|
2023-03-03 13:45:38 +00:00
|
|
|
|
|
|
|
argTypes.push_back(argTy);
|
|
|
|
argNames.emplace_back(FunctionArgument{local->name.value, local->location});
|
2023-10-13 20:38:31 +01:00
|
|
|
|
2023-11-17 18:15:31 +00:00
|
|
|
signatureScope->bindings[local] = Binding{argTy, local->location};
|
2023-03-03 13:45:38 +00:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
DefId def = dfg->getDef(local);
|
|
|
|
signatureScope->lvalueTypes[def] = argTy;
|
|
|
|
signatureScope->rvalueRefinements[def] = argTy;
|
2022-07-01 00:29:02 +01:00
|
|
|
}
|
|
|
|
|
2022-09-15 23:13:58 +01:00
|
|
|
TypePackId varargPack = nullptr;
|
2022-09-08 22:44:50 +01:00
|
|
|
|
|
|
|
if (fn->vararg)
|
|
|
|
{
|
|
|
|
if (fn->varargAnnotation)
|
|
|
|
{
|
2023-03-03 13:45:38 +00:00
|
|
|
TypePackId annotationType =
|
|
|
|
resolveTypePack(signatureScope, fn->varargAnnotation, /* inTypeArguments */ false, /* replaceErrorWithFresh */ true);
|
2022-09-08 22:44:50 +01:00
|
|
|
varargPack = annotationType;
|
|
|
|
}
|
2022-12-02 10:46:05 +00:00
|
|
|
else if (expectedArgPack.tail && get<VariadicTypePack>(*expectedArgPack.tail))
|
|
|
|
varargPack = *expectedArgPack.tail;
|
2022-09-08 22:44:50 +01:00
|
|
|
else
|
2023-01-03 17:33:19 +00:00
|
|
|
varargPack = builtinTypes->anyTypePack;
|
2022-09-08 22:44:50 +01:00
|
|
|
|
|
|
|
signatureScope->varargPack = varargPack;
|
2022-12-02 10:46:05 +00:00
|
|
|
bodyScope->varargPack = varargPack;
|
2022-09-08 22:44:50 +01:00
|
|
|
}
|
2022-09-15 23:13:58 +01:00
|
|
|
else
|
|
|
|
{
|
2023-01-03 17:33:19 +00:00
|
|
|
varargPack = arena->addTypePack(VariadicTypePack{builtinTypes->anyType, /*hidden*/ true});
|
2022-09-15 23:13:58 +01:00
|
|
|
// We do not add to signatureScope->varargPack because ... is not valid
|
|
|
|
// in functions without an explicit ellipsis.
|
2022-12-02 10:46:05 +00:00
|
|
|
|
|
|
|
signatureScope->varargPack = std::nullopt;
|
|
|
|
bodyScope->varargPack = std::nullopt;
|
2022-09-15 23:13:58 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
LUAU_ASSERT(nullptr != varargPack);
|
2022-09-08 22:44:50 +01:00
|
|
|
|
2022-12-02 10:46:05 +00:00
|
|
|
// If there is both an annotation and an expected type, the annotation wins.
|
|
|
|
// Type checking will sort out any discrepancies later.
|
2022-06-24 02:44:07 +01:00
|
|
|
if (fn->returnAnnotation)
|
|
|
|
{
|
2023-03-03 13:45:38 +00:00
|
|
|
TypePackId annotatedRetType =
|
|
|
|
resolveTypePack(signatureScope, *fn->returnAnnotation, /* inTypeArguments */ false, /* replaceErrorWithFresh*/ true);
|
2022-09-23 19:32:10 +01:00
|
|
|
// 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<FreeTypePack>(returnType));
|
|
|
|
asMutable(returnType)->ty.emplace<BoundTypePack>(annotatedRetType);
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
2022-12-02 10:46:05 +00:00
|
|
|
else if (expectedFunction)
|
2022-06-03 21:32:20 +01:00
|
|
|
{
|
2022-12-02 10:46:05 +00:00
|
|
|
asMutable(returnType)->ty.emplace<BoundTypePack>(expectedFunction->retTypes);
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
|
2022-07-01 00:29:02 +01:00
|
|
|
// TODO: Preserve argument names in the function's type.
|
2022-06-24 02:44:07 +01:00
|
|
|
|
2023-01-03 17:33:19 +00:00
|
|
|
FunctionType actualFunction{TypeLevel{}, parent.get(), arena->addTypePack(argTypes, varargPack), returnType};
|
2022-07-01 00:29:02 +01:00
|
|
|
actualFunction.generics = std::move(genericTypes);
|
|
|
|
actualFunction.genericPacks = std::move(genericTypePacks);
|
2023-02-17 14:53:37 +00:00
|
|
|
actualFunction.argNames = std::move(argNames);
|
2023-02-24 18:24:22 +00:00
|
|
|
actualFunction.hasSelf = fn->self != nullptr;
|
2022-07-01 00:29:02 +01:00
|
|
|
|
2023-05-19 19:59:59 +01:00
|
|
|
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;
|
|
|
|
|
2022-06-17 01:54:42 +01:00
|
|
|
TypeId actualFunctionType = arena->addType(std::move(actualFunction));
|
|
|
|
LUAU_ASSERT(actualFunctionType);
|
2023-02-03 12:34:12 +00:00
|
|
|
module->astTypes[fn] = actualFunctionType;
|
2022-06-17 01:54:42 +01:00
|
|
|
|
2023-01-03 17:33:19 +00:00
|
|
|
if (expectedType && get<FreeType>(*expectedType))
|
2023-02-24 18:24:22 +00:00
|
|
|
bindFreeType(*expectedType, actualFunctionType);
|
2022-12-02 10:46:05 +00:00
|
|
|
|
2022-07-01 00:29:02 +01:00
|
|
|
return {
|
|
|
|
/* signature */ actualFunctionType,
|
2022-12-02 10:46:05 +00:00
|
|
|
/* signatureScope */ signatureScope,
|
2022-07-29 04:41:13 +01:00
|
|
|
/* bodyScope */ bodyScope,
|
2022-07-01 00:29:02 +01:00
|
|
|
};
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
void ConstraintGenerator::checkFunctionBody(const ScopePtr& scope, AstExprFunction* fn)
|
2022-06-17 01:54:42 +01:00
|
|
|
{
|
|
|
|
// If it is possible for execution to reach the end of the function, the return type must be compatible with ()
|
2024-02-23 18:40:00 +00:00
|
|
|
ControlFlow cf = visitBlockWithoutChildScope(scope, fn->body);
|
|
|
|
if (cf == ControlFlow::None)
|
|
|
|
addConstraint(scope, fn->location, PackSubtypeConstraint{builtinTypes->emptyTypePack, scope->returnType});
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
TypeId ConstraintGenerator::resolveType(const ScopePtr& scope, AstType* ty, bool inTypeArguments, bool replaceErrorWithFresh)
|
2022-06-24 02:44:07 +01:00
|
|
|
{
|
|
|
|
TypeId result = nullptr;
|
|
|
|
|
|
|
|
if (auto ref = ty->as<AstTypeReference>())
|
|
|
|
{
|
2022-09-29 23:11:54 +01:00
|
|
|
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"});
|
2023-05-19 19:59:59 +01:00
|
|
|
module->astResolvedTypes[ty] = builtinTypes->errorRecoveryType();
|
2023-01-03 17:33:19 +00:00
|
|
|
return builtinTypes->errorRecoveryType();
|
2022-09-29 23:11:54 +01:00
|
|
|
}
|
|
|
|
else
|
2023-01-20 12:02:39 +00:00
|
|
|
return resolveType(scope, ref->parameters.data[0].type, inTypeArguments);
|
2022-09-29 23:11:54 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-27 23:22:49 +01:00
|
|
|
std::optional<TypeFun> alias;
|
2022-08-04 22:27:28 +01:00
|
|
|
|
2022-10-27 23:22:49 +01:00
|
|
|
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())
|
2022-08-04 22:27:28 +01:00
|
|
|
{
|
|
|
|
// If the alias is not generic, we don't need to set up a blocked
|
|
|
|
// type and an instantiation constraint.
|
2022-09-02 00:00:14 +01:00
|
|
|
if (alias.has_value() && alias->typeParams.empty() && alias->typePackParams.empty())
|
2022-08-04 22:27:28 +01:00
|
|
|
{
|
|
|
|
result = alias->type;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
std::vector<TypeId> parameters;
|
|
|
|
std::vector<TypePackId> 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)
|
|
|
|
{
|
2023-01-20 12:02:39 +00:00
|
|
|
parameters.push_back(resolveType(scope, p.type, /* inTypeArguments */ true));
|
2022-08-04 22:27:28 +01:00
|
|
|
}
|
|
|
|
else if (p.typePack)
|
|
|
|
{
|
2023-01-20 12:02:39 +00:00
|
|
|
packParameters.push_back(resolveTypePack(scope, p.typePack, /* inTypeArguments */ true));
|
2022-08-04 22:27:28 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// This indicates a parser bug: one of these two pointers
|
|
|
|
// should be set.
|
|
|
|
LUAU_ASSERT(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-03 17:33:19 +00:00
|
|
|
result = arena->addType(PendingExpansionType{ref->prefix, ref->name, parameters, packParameters});
|
2022-08-04 22:27:28 +01:00
|
|
|
|
2023-01-20 12:02:39 +00:00
|
|
|
// 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)
|
2022-09-02 00:00:14 +01:00
|
|
|
addConstraint(scope, ty->location, TypeAliasExpansionConstraint{/* target */ result});
|
2022-08-04 22:27:28 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-01-03 17:33:19 +00:00
|
|
|
result = builtinTypes->errorRecoveryType();
|
2023-03-03 13:45:38 +00:00
|
|
|
if (replaceErrorWithFresh)
|
|
|
|
result = freshType(scope);
|
2022-08-04 22:27:28 +01:00
|
|
|
}
|
2022-06-03 21:32:20 +01:00
|
|
|
}
|
2022-06-24 02:44:07 +01:00
|
|
|
else if (auto tab = ty->as<AstTypeTable>())
|
|
|
|
{
|
2023-01-03 17:33:19 +00:00
|
|
|
TableType::Props props;
|
2022-06-24 02:44:07 +01:00
|
|
|
std::optional<TableIndexer> indexer;
|
|
|
|
|
|
|
|
for (const AstTableProp& prop : tab->props)
|
|
|
|
{
|
2024-02-23 18:40:00 +00:00
|
|
|
// TODO: Recursion limit.
|
|
|
|
TypeId propTy = resolveType(scope, prop.type, inTypeArguments);
|
|
|
|
|
|
|
|
Property& p = props[prop.name.value];
|
|
|
|
p.typeLocation = prop.location;
|
|
|
|
|
|
|
|
switch (prop.access)
|
2024-01-19 15:13:08 +00:00
|
|
|
{
|
2024-02-23 18:40:00 +00:00
|
|
|
case AstTableAccess::ReadWrite:
|
|
|
|
p.readTy = propTy;
|
|
|
|
p.writeTy = propTy;
|
|
|
|
break;
|
|
|
|
case AstTableAccess::Read:
|
|
|
|
p.readTy = propTy;
|
|
|
|
break;
|
|
|
|
case AstTableAccess::Write:
|
|
|
|
reportError(*prop.accessLocation, GenericError{"write keyword is illegal here"});
|
|
|
|
p.readTy = propTy;
|
|
|
|
p.writeTy = propTy;
|
|
|
|
break;
|
|
|
|
default:
|
2024-01-19 15:13:08 +00:00
|
|
|
ice->ice("Unexpected property access " + std::to_string(int(prop.access)));
|
2024-02-23 18:40:00 +00:00
|
|
|
break;
|
|
|
|
}
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
|
2024-01-19 15:13:08 +00:00
|
|
|
if (AstTableIndexer* astIndexer = tab->indexer)
|
2022-06-24 02:44:07 +01:00
|
|
|
{
|
2024-01-19 15:13:08 +00:00
|
|
|
if (astIndexer->access == AstTableAccess::Read)
|
|
|
|
reportError(astIndexer->accessLocation.value_or(Location{}), GenericError{"read keyword is illegal here"});
|
|
|
|
else if (astIndexer->access == AstTableAccess::Write)
|
|
|
|
reportError(astIndexer->accessLocation.value_or(Location{}), GenericError{"write keyword is illegal here"});
|
|
|
|
else if (astIndexer->access == AstTableAccess::ReadWrite)
|
|
|
|
{
|
|
|
|
// TODO: Recursion limit.
|
|
|
|
indexer = TableIndexer{
|
|
|
|
resolveType(scope, astIndexer->indexType, inTypeArguments),
|
|
|
|
resolveType(scope, astIndexer->resultType, inTypeArguments),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
else
|
|
|
|
ice->ice("Unexpected property access " + std::to_string(int(astIndexer->access)));
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
|
2023-01-03 17:33:19 +00:00
|
|
|
result = arena->addType(TableType{props, indexer, scope->level, scope.get(), TableState::Sealed});
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
else if (auto fn = ty->as<AstTypeFunction>())
|
|
|
|
{
|
|
|
|
// TODO: Recursion limit.
|
2022-07-01 00:29:02 +01:00
|
|
|
bool hasGenerics = fn->generics.size > 0 || fn->genericPacks.size > 0;
|
2022-07-29 04:41:13 +01:00
|
|
|
ScopePtr signatureScope = nullptr;
|
2022-07-01 00:29:02 +01:00
|
|
|
|
|
|
|
std::vector<TypeId> genericTypes;
|
|
|
|
std::vector<TypePackId> genericTypePacks;
|
2022-06-24 02:44:07 +01:00
|
|
|
|
2022-07-01 00:29:02 +01:00
|
|
|
// If we don't have generics, we do not need to generate a child scope
|
|
|
|
// for the generic bindings to live on.
|
|
|
|
if (hasGenerics)
|
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
signatureScope = childScope(fn, scope);
|
2022-07-01 00:29:02 +01:00
|
|
|
|
2022-07-29 04:41:13 +01:00
|
|
|
std::vector<std::pair<Name, GenericTypeDefinition>> genericDefinitions = createGenerics(signatureScope, fn->generics);
|
|
|
|
std::vector<std::pair<Name, GenericTypePackDefinition>> genericPackDefinitions = createGenericPacks(signatureScope, fn->genericPacks);
|
2022-06-24 02:44:07 +01:00
|
|
|
|
2022-07-01 00:29:02 +01:00
|
|
|
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.
|
2022-07-29 04:41:13 +01:00
|
|
|
signatureScope = scope;
|
2022-07-01 00:29:02 +01:00
|
|
|
}
|
|
|
|
|
2023-03-03 13:45:38 +00:00
|
|
|
TypePackId argTypes = resolveTypePack(signatureScope, fn->argTypes, inTypeArguments, replaceErrorWithFresh);
|
|
|
|
TypePackId returnTypes = resolveTypePack(signatureScope, fn->returnTypes, inTypeArguments, replaceErrorWithFresh);
|
2022-07-01 00:29:02 +01:00
|
|
|
|
2023-01-03 17:33:19 +00:00
|
|
|
// TODO: FunctionType needs a pointer to the scope so that we know
|
2022-07-01 00:29:02 +01:00
|
|
|
// how to quantify/instantiate it.
|
2023-01-03 17:33:19 +00:00
|
|
|
FunctionType ftv{TypeLevel{}, scope.get(), {}, {}, argTypes, returnTypes};
|
2023-10-06 18:31:16 +01:00
|
|
|
ftv.isCheckedFunction = fn->checkedFunction;
|
2022-07-01 00:29:02 +01:00
|
|
|
|
2023-01-03 17:33:19 +00:00
|
|
|
// This replicates the behavior of the appropriate FunctionType
|
2022-07-01 00:29:02 +01:00
|
|
|
// constructors.
|
|
|
|
ftv.generics = std::move(genericTypes);
|
|
|
|
ftv.genericPacks = std::move(genericTypePacks);
|
|
|
|
|
|
|
|
ftv.argNames.reserve(fn->argNames.size);
|
2022-06-24 02:44:07 +01:00
|
|
|
for (const auto& el : fn->argNames)
|
|
|
|
{
|
|
|
|
if (el)
|
|
|
|
{
|
|
|
|
const auto& [name, location] = *el;
|
2022-07-01 00:29:02 +01:00
|
|
|
ftv.argNames.push_back(FunctionArgument{name.value, location});
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-07-01 00:29:02 +01:00
|
|
|
ftv.argNames.push_back(std::nullopt);
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
}
|
2022-07-01 00:29:02 +01:00
|
|
|
|
|
|
|
result = arena->addType(std::move(ftv));
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
else if (auto tof = ty->as<AstTypeTypeof>())
|
|
|
|
{
|
|
|
|
// TODO: Recursion limit.
|
2022-10-27 23:22:49 +01:00
|
|
|
TypeId exprType = check(scope, tof->expr).ty;
|
2022-06-24 02:44:07 +01:00
|
|
|
result = exprType;
|
|
|
|
}
|
|
|
|
else if (auto unionAnnotation = ty->as<AstTypeUnion>())
|
|
|
|
{
|
|
|
|
std::vector<TypeId> parts;
|
|
|
|
for (AstType* part : unionAnnotation->types)
|
|
|
|
{
|
|
|
|
// TODO: Recursion limit.
|
2023-01-20 12:02:39 +00:00
|
|
|
parts.push_back(resolveType(scope, part, inTypeArguments));
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
|
2023-01-03 17:33:19 +00:00
|
|
|
result = arena->addType(UnionType{parts});
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
else if (auto intersectionAnnotation = ty->as<AstTypeIntersection>())
|
|
|
|
{
|
|
|
|
std::vector<TypeId> parts;
|
|
|
|
for (AstType* part : intersectionAnnotation->types)
|
|
|
|
{
|
|
|
|
// TODO: Recursion limit.
|
2023-01-20 12:02:39 +00:00
|
|
|
parts.push_back(resolveType(scope, part, inTypeArguments));
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
|
2023-01-03 17:33:19 +00:00
|
|
|
result = arena->addType(IntersectionType{parts});
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
else if (auto boolAnnotation = ty->as<AstTypeSingletonBool>())
|
|
|
|
{
|
2023-10-20 21:36:26 +01:00
|
|
|
if (boolAnnotation->value)
|
|
|
|
result = builtinTypes->trueType;
|
|
|
|
else
|
|
|
|
result = builtinTypes->falseType;
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
else if (auto stringAnnotation = ty->as<AstTypeSingletonString>())
|
|
|
|
{
|
2023-01-03 17:33:19 +00:00
|
|
|
result = arena->addType(SingletonType(StringSingleton{std::string(stringAnnotation->value.data, stringAnnotation->value.size)}));
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
else if (ty->is<AstTypeError>())
|
|
|
|
{
|
2023-01-03 17:33:19 +00:00
|
|
|
result = builtinTypes->errorRecoveryType();
|
2023-03-03 13:45:38 +00:00
|
|
|
if (replaceErrorWithFresh)
|
|
|
|
result = freshType(scope);
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(0);
|
2023-01-03 17:33:19 +00:00
|
|
|
result = builtinTypes->errorRecoveryType();
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
|
2023-02-03 12:34:12 +00:00
|
|
|
module->astResolvedTypes[ty] = result;
|
2022-06-24 02:44:07 +01:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
TypePackId ConstraintGenerator::resolveTypePack(const ScopePtr& scope, AstTypePack* tp, bool inTypeArgument, bool replaceErrorWithFresh)
|
2022-06-24 02:44:07 +01:00
|
|
|
{
|
|
|
|
TypePackId result;
|
|
|
|
if (auto expl = tp->as<AstTypePackExplicit>())
|
|
|
|
{
|
2023-03-03 13:45:38 +00:00
|
|
|
result = resolveTypePack(scope, expl->typeList, inTypeArgument, replaceErrorWithFresh);
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
else if (auto var = tp->as<AstTypePackVariadic>())
|
|
|
|
{
|
2023-03-03 13:45:38 +00:00
|
|
|
TypeId ty = resolveType(scope, var->variadicType, inTypeArgument, replaceErrorWithFresh);
|
2022-06-24 02:44:07 +01:00
|
|
|
result = arena->addTypePack(TypePackVar{VariadicTypePack{ty}});
|
|
|
|
}
|
|
|
|
else if (auto gen = tp->as<AstTypePackGeneric>())
|
|
|
|
{
|
2022-08-11 21:42:54 +01:00
|
|
|
if (std::optional<TypePackId> lookup = scope->lookupPack(gen->genericName.value))
|
2022-08-04 22:27:28 +01:00
|
|
|
{
|
|
|
|
result = *lookup;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
reportError(tp->location, UnknownSymbol{gen->genericName.value, UnknownSymbol::Context::Type});
|
2023-01-03 17:33:19 +00:00
|
|
|
result = builtinTypes->errorRecoveryTypePack();
|
2022-08-04 22:27:28 +01:00
|
|
|
}
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(0);
|
2023-01-03 17:33:19 +00:00
|
|
|
result = builtinTypes->errorRecoveryTypePack();
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
|
2023-02-03 12:34:12 +00:00
|
|
|
module->astResolvedTypePacks[tp] = result;
|
2022-06-24 02:44:07 +01:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
TypePackId ConstraintGenerator::resolveTypePack(const ScopePtr& scope, const AstTypeList& list, bool inTypeArguments, bool replaceErrorWithFresh)
|
2022-06-24 02:44:07 +01:00
|
|
|
{
|
|
|
|
std::vector<TypeId> head;
|
|
|
|
|
|
|
|
for (AstType* headTy : list.types)
|
|
|
|
{
|
2023-03-03 13:45:38 +00:00
|
|
|
head.push_back(resolveType(scope, headTy, inTypeArguments, replaceErrorWithFresh));
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<TypePackId> tail = std::nullopt;
|
|
|
|
if (list.tailType)
|
|
|
|
{
|
2023-03-03 13:45:38 +00:00
|
|
|
tail = resolveTypePack(scope, list.tailType, inTypeArguments, replaceErrorWithFresh);
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
|
2024-02-02 18:20:03 +00:00
|
|
|
return addTypePack(std::move(head), tail);
|
2022-06-03 21:32:20 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
std::vector<std::pair<Name, GenericTypeDefinition>> ConstraintGenerator::createGenerics(
|
2023-02-24 18:24:22 +00:00
|
|
|
const ScopePtr& scope, AstArray<AstGenericType> generics, bool useCache, bool addTypes)
|
2022-07-01 00:29:02 +01:00
|
|
|
{
|
|
|
|
std::vector<std::pair<Name, GenericTypeDefinition>> result;
|
|
|
|
for (const auto& generic : generics)
|
|
|
|
{
|
2023-02-17 14:53:37 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2022-07-01 00:29:02 +01:00
|
|
|
std::optional<TypeId> defaultTy = std::nullopt;
|
|
|
|
|
|
|
|
if (generic.defaultValue)
|
2023-01-20 12:02:39 +00:00
|
|
|
defaultTy = resolveType(scope, generic.defaultValue, /* inTypeArguments */ false);
|
2022-07-01 00:29:02 +01:00
|
|
|
|
2023-02-24 18:24:22 +00:00
|
|
|
if (addTypes)
|
|
|
|
scope->privateTypeBindings[generic.name.value] = TypeFun{genericTy};
|
|
|
|
|
2022-10-27 23:22:49 +01:00
|
|
|
result.push_back({generic.name.value, GenericTypeDefinition{genericTy, defaultTy}});
|
2022-07-01 00:29:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
std::vector<std::pair<Name, GenericTypePackDefinition>> ConstraintGenerator::createGenericPacks(
|
2023-02-24 18:24:22 +00:00
|
|
|
const ScopePtr& scope, AstArray<AstGenericTypePack> generics, bool useCache, bool addTypes)
|
2022-07-01 00:29:02 +01:00
|
|
|
{
|
|
|
|
std::vector<std::pair<Name, GenericTypePackDefinition>> result;
|
|
|
|
for (const auto& generic : generics)
|
|
|
|
{
|
2023-02-17 14:53:37 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2022-07-01 00:29:02 +01:00
|
|
|
std::optional<TypePackId> defaultTy = std::nullopt;
|
|
|
|
|
|
|
|
if (generic.defaultValue)
|
2023-01-20 12:02:39 +00:00
|
|
|
defaultTy = resolveTypePack(scope, generic.defaultValue, /* inTypeArguments */ false);
|
2022-07-01 00:29:02 +01:00
|
|
|
|
2023-02-24 18:24:22 +00:00
|
|
|
if (addTypes)
|
|
|
|
scope->privateTypePackBindings[generic.name.value] = genericTy;
|
|
|
|
|
2022-10-27 23:22:49 +01:00
|
|
|
result.push_back({generic.name.value, GenericTypePackDefinition{genericTy, defaultTy}});
|
2022-07-01 00:29:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
Inference ConstraintGenerator::flattenPack(const ScopePtr& scope, Location location, InferencePack pack)
|
2022-07-01 00:29:02 +01:00
|
|
|
{
|
2023-02-03 12:34:12 +00:00
|
|
|
const auto& [tp, refinements] = pack;
|
|
|
|
RefinementId refinement = nullptr;
|
|
|
|
if (!refinements.empty())
|
|
|
|
refinement = refinements[0];
|
2022-11-04 17:02:37 +00:00
|
|
|
|
2022-07-01 00:29:02 +01:00
|
|
|
if (auto f = first(tp))
|
2023-02-03 12:34:12 +00:00
|
|
|
return Inference{*f, refinement};
|
2022-07-01 00:29:02 +01:00
|
|
|
|
2023-02-24 18:24:22 +00:00
|
|
|
TypeId typeResult = arena->addType(BlockedType{});
|
|
|
|
TypePackId resultPack = arena->addTypePack({typeResult}, arena->freshTypePack(scope.get()));
|
|
|
|
addConstraint(scope, location, UnpackConstraint{resultPack, tp});
|
2022-07-01 00:29:02 +01:00
|
|
|
|
2023-02-03 12:34:12 +00:00
|
|
|
return Inference{typeResult, refinement};
|
2022-07-01 00:29:02 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
void ConstraintGenerator::reportError(Location location, TypeErrorData err)
|
2022-07-01 00:29:02 +01:00
|
|
|
{
|
2023-04-21 22:41:03 +01:00
|
|
|
errors.push_back(TypeError{location, module->name, std::move(err)});
|
2022-09-08 22:44:50 +01:00
|
|
|
|
2023-03-03 13:45:38 +00:00
|
|
|
if (logger)
|
2022-09-08 22:44:50 +01:00
|
|
|
logger->captureGenerationError(errors.back());
|
2022-07-01 00:29:02 +01:00
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
void ConstraintGenerator::reportCodeTooComplex(Location location)
|
2022-07-01 00:29:02 +01:00
|
|
|
{
|
2023-04-21 22:41:03 +01:00
|
|
|
errors.push_back(TypeError{location, module->name, CodeTooComplex{}});
|
2022-09-08 22:44:50 +01:00
|
|
|
|
2023-03-03 13:45:38 +00:00
|
|
|
if (logger)
|
2022-09-08 22:44:50 +01:00
|
|
|
logger->captureGenerationError(errors.back());
|
2022-07-01 00:29:02 +01:00
|
|
|
}
|
|
|
|
|
2024-01-27 02:30:40 +00:00
|
|
|
TypeId ConstraintGenerator::makeUnion(const ScopePtr& scope, Location location, TypeId lhs, TypeId rhs)
|
|
|
|
{
|
|
|
|
TypeId resultType = arena->addType(TypeFamilyInstanceType{
|
|
|
|
NotNull{&kBuiltinTypeFamilies.unionFamily},
|
|
|
|
{lhs, rhs},
|
|
|
|
{},
|
|
|
|
});
|
|
|
|
addConstraint(scope, location, ReduceConstraint{resultType});
|
|
|
|
|
|
|
|
return resultType;
|
|
|
|
}
|
|
|
|
|
|
|
|
TypeId ConstraintGenerator::makeIntersect(const ScopePtr& scope, Location location, TypeId lhs, TypeId rhs)
|
|
|
|
{
|
|
|
|
TypeId resultType = arena->addType(TypeFamilyInstanceType{
|
|
|
|
NotNull{&kBuiltinTypeFamilies.intersectFamily},
|
|
|
|
{lhs, rhs},
|
|
|
|
{},
|
|
|
|
});
|
|
|
|
addConstraint(scope, location, ReduceConstraint{resultType});
|
|
|
|
|
|
|
|
return resultType;
|
|
|
|
}
|
|
|
|
|
2022-07-01 00:29:02 +01:00
|
|
|
struct GlobalPrepopulator : AstVisitor
|
|
|
|
{
|
2022-07-29 04:41:13 +01:00
|
|
|
const NotNull<Scope> globalScope;
|
2022-07-01 00:29:02 +01:00
|
|
|
const NotNull<TypeArena> arena;
|
2023-10-06 18:31:16 +01:00
|
|
|
const NotNull<const DataFlowGraph> dfg;
|
2022-07-01 00:29:02 +01:00
|
|
|
|
2023-10-06 18:31:16 +01:00
|
|
|
GlobalPrepopulator(NotNull<Scope> globalScope, NotNull<TypeArena> arena, NotNull<const DataFlowGraph> dfg)
|
2022-07-01 00:29:02 +01:00
|
|
|
: globalScope(globalScope)
|
|
|
|
, arena(arena)
|
2023-10-06 18:31:16 +01:00
|
|
|
, dfg(dfg)
|
2022-07-01 00:29:02 +01:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2023-12-02 02:04:44 +00:00
|
|
|
bool visit(AstExprGlobal* global) override
|
|
|
|
{
|
|
|
|
if (auto ty = globalScope->lookup(global->name))
|
|
|
|
{
|
|
|
|
DefId def = dfg->getDef(global);
|
|
|
|
globalScope->lvalueTypes[def] = *ty;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-07-01 00:29:02 +01:00
|
|
|
bool visit(AstStatFunction* function) override
|
|
|
|
{
|
|
|
|
if (AstExprGlobal* g = function->name->as<AstExprGlobal>())
|
2023-10-06 18:31:16 +01:00
|
|
|
{
|
|
|
|
TypeId bt = arena->addType(BlockedType{});
|
|
|
|
globalScope->bindings[g->name] = Binding{bt};
|
|
|
|
}
|
2022-07-01 00:29:02 +01:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2023-12-08 15:42:54 +00:00
|
|
|
|
|
|
|
bool visit(AstType*) override
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(class AstTypePack* node) override
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
2022-07-01 00:29:02 +01:00
|
|
|
};
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
void ConstraintGenerator::prepopulateGlobalScope(const ScopePtr& globalScope, AstStatBlock* program)
|
2022-07-01 00:29:02 +01:00
|
|
|
{
|
2023-10-06 18:31:16 +01:00
|
|
|
GlobalPrepopulator gp{NotNull{globalScope.get()}, arena, dfg};
|
2022-07-01 00:29:02 +01:00
|
|
|
|
2023-04-28 12:55:55 +01:00
|
|
|
if (prepareModuleScope)
|
|
|
|
prepareModuleScope(module->name, globalScope);
|
|
|
|
|
2022-07-01 00:29:02 +01:00
|
|
|
program->visit(&gp);
|
|
|
|
}
|
|
|
|
|
2023-11-17 18:15:31 +00:00
|
|
|
void ConstraintGenerator::recordInferredBinding(AstLocal* local, TypeId ty)
|
|
|
|
{
|
|
|
|
if (InferredBinding* ib = inferredBindings.find(local))
|
|
|
|
ib->types.insert(ty);
|
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
void ConstraintGenerator::fillInInferredBindings(const ScopePtr& globalScope, AstStatBlock* block)
|
2023-10-13 20:38:31 +01:00
|
|
|
{
|
2023-10-20 21:36:26 +01:00
|
|
|
for (const auto& [symbol, p] : inferredBindings)
|
2023-10-13 20:38:31 +01:00
|
|
|
{
|
2023-10-20 21:36:26 +01:00
|
|
|
const auto& [scope, location, types] = p;
|
2023-10-13 20:38:31 +01:00
|
|
|
|
2023-10-20 21:36:26 +01:00
|
|
|
std::vector<TypeId> tys(types.begin(), types.end());
|
2024-01-19 15:13:08 +00:00
|
|
|
if (tys.size() == 1)
|
|
|
|
scope->bindings[symbol] = Binding{tys.front(), location};
|
|
|
|
else
|
|
|
|
{
|
|
|
|
TypeId ty = arena->addType(BlockedType{});
|
|
|
|
addConstraint(globalScope, Location{}, SetOpConstraint{SetOpConstraint::Union, ty, std::move(tys)});
|
2023-10-13 20:38:31 +01:00
|
|
|
|
2024-01-19 15:13:08 +00:00
|
|
|
scope->bindings[symbol] = Binding{ty, location};
|
|
|
|
}
|
2023-10-13 20:38:31 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-03 19:47:28 +00:00
|
|
|
std::vector<std::optional<TypeId>> ConstraintGenerator::getExpectedCallTypesForFunctionOverloads(const TypeId fnType)
|
2023-03-03 13:45:38 +00:00
|
|
|
{
|
|
|
|
std::vector<TypeId> funTys;
|
|
|
|
if (auto it = get<IntersectionType>(follow(fnType)))
|
|
|
|
{
|
|
|
|
for (TypeId intersectionComponent : it)
|
|
|
|
{
|
|
|
|
funTys.push_back(intersectionComponent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<std::optional<TypeId>> 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<TypeId> 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<FunctionType>(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<VariadicTypePack>(*argsTail))
|
|
|
|
{
|
|
|
|
while (index < funTys.size())
|
|
|
|
assignOption(index++, vtp->ty);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO vvijay Feb 24, 2023 apparently we have to demote the types here?
|
|
|
|
|
|
|
|
return expectedTypes;
|
|
|
|
}
|
|
|
|
|
2022-11-10 22:04:44 +00:00
|
|
|
std::vector<NotNull<Constraint>> borrowConstraints(const std::vector<ConstraintPtr>& constraints)
|
2022-06-03 21:32:20 +01:00
|
|
|
{
|
2022-11-10 22:04:44 +00:00
|
|
|
std::vector<NotNull<Constraint>> result;
|
|
|
|
result.reserve(constraints.size());
|
2022-06-03 21:32:20 +01:00
|
|
|
|
2022-11-10 22:04:44 +00:00
|
|
|
for (const auto& c : constraints)
|
|
|
|
result.emplace_back(c.get());
|
2022-06-03 21:32:20 +01:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace Luau
|