2022-09-08 22:44:50 +01:00
|
|
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
2022-06-17 01:54:42 +01:00
|
|
|
#include "Luau/TypeChecker2.h"
|
|
|
|
|
|
|
|
#include "Luau/Ast.h"
|
|
|
|
#include "Luau/AstQuery.h"
|
|
|
|
#include "Luau/Clone.h"
|
2022-07-01 00:29:02 +01:00
|
|
|
#include "Luau/Instantiation.h"
|
2022-06-17 01:54:42 +01:00
|
|
|
#include "Luau/Normalize.h"
|
2022-08-18 22:04:33 +01:00
|
|
|
#include "Luau/ToString.h"
|
2022-07-01 00:29:02 +01:00
|
|
|
#include "Luau/TxnLog.h"
|
|
|
|
#include "Luau/TypeUtils.h"
|
2022-08-11 21:42:54 +01:00
|
|
|
#include "Luau/TypeVar.h"
|
2022-06-24 02:44:07 +01:00
|
|
|
#include "Luau/Unifier.h"
|
2022-09-08 22:44:50 +01:00
|
|
|
#include "Luau/ToString.h"
|
|
|
|
#include "Luau/DcrLogger.h"
|
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
|
|
|
|
LUAU_FASTFLAG(DebugLuauLogSolverToJson);
|
2022-06-17 01:54:42 +01:00
|
|
|
|
|
|
|
namespace Luau
|
|
|
|
{
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
/* Push a scope onto the end of a stack for the lifetime of the StackPusher instance.
|
|
|
|
* TypeChecker2 uses this to maintain knowledge about which scope encloses every
|
|
|
|
* given AstNode.
|
|
|
|
*/
|
|
|
|
struct StackPusher
|
|
|
|
{
|
|
|
|
std::vector<NotNull<Scope>>* stack;
|
|
|
|
NotNull<Scope> scope;
|
|
|
|
|
|
|
|
explicit StackPusher(std::vector<NotNull<Scope>>& stack, Scope* scope)
|
|
|
|
: stack(&stack)
|
|
|
|
, scope(scope)
|
|
|
|
{
|
|
|
|
stack.push_back(NotNull{scope});
|
|
|
|
}
|
|
|
|
|
|
|
|
~StackPusher()
|
|
|
|
{
|
|
|
|
if (stack)
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(stack->back() == scope);
|
|
|
|
stack->pop_back();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
StackPusher(const StackPusher&) = delete;
|
|
|
|
StackPusher&& operator=(const StackPusher&) = delete;
|
|
|
|
|
|
|
|
StackPusher(StackPusher&& other)
|
|
|
|
: stack(std::exchange(other.stack, nullptr))
|
|
|
|
, scope(other.scope)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct TypeChecker2
|
2022-06-17 01:54:42 +01:00
|
|
|
{
|
2022-09-08 22:44:50 +01:00
|
|
|
NotNull<SingletonTypes> singletonTypes;
|
|
|
|
DcrLogger* logger;
|
|
|
|
InternalErrorReporter ice; // FIXME accept a pointer from Frontend
|
2022-06-17 01:54:42 +01:00
|
|
|
const SourceModule* sourceModule;
|
|
|
|
Module* module;
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
std::vector<NotNull<Scope>> stack;
|
|
|
|
|
2022-09-08 22:44:50 +01:00
|
|
|
TypeChecker2(NotNull<SingletonTypes> singletonTypes, DcrLogger* logger, const SourceModule* sourceModule, Module* module)
|
|
|
|
: singletonTypes(singletonTypes)
|
|
|
|
, logger(logger)
|
|
|
|
, sourceModule(sourceModule)
|
2022-06-17 01:54:42 +01:00
|
|
|
, module(module)
|
|
|
|
{
|
2022-09-08 22:44:50 +01:00
|
|
|
if (FFlag::DebugLuauLogSolverToJson)
|
|
|
|
LUAU_ASSERT(logger);
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
std::optional<StackPusher> pushStack(AstNode* node)
|
|
|
|
{
|
|
|
|
if (Scope** scope = module->astScopes.find(node))
|
|
|
|
return StackPusher{stack, *scope};
|
|
|
|
else
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
2022-06-17 01:54:42 +01:00
|
|
|
|
|
|
|
TypePackId lookupPack(AstExpr* expr)
|
|
|
|
{
|
2022-07-01 00:29:02 +01:00
|
|
|
// If a type isn't in the type graph, it probably means that a recursion limit was exceeded.
|
|
|
|
// We'll just return anyType in these cases. Typechecking against any is very fast and this
|
|
|
|
// allows us not to think about this very much in the actual typechecking logic.
|
2022-06-17 01:54:42 +01:00
|
|
|
TypePackId* tp = module->astTypePacks.find(expr);
|
2022-07-01 00:29:02 +01:00
|
|
|
if (tp)
|
|
|
|
return follow(*tp);
|
|
|
|
else
|
2022-09-08 22:44:50 +01:00
|
|
|
return singletonTypes->anyTypePack;
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
TypeId lookupType(AstExpr* expr)
|
|
|
|
{
|
2022-07-01 00:29:02 +01:00
|
|
|
// If a type isn't in the type graph, it probably means that a recursion limit was exceeded.
|
|
|
|
// We'll just return anyType in these cases. Typechecking against any is very fast and this
|
|
|
|
// allows us not to think about this very much in the actual typechecking logic.
|
2022-06-17 01:54:42 +01:00
|
|
|
TypeId* ty = module->astTypes.find(expr);
|
2022-07-01 00:29:02 +01:00
|
|
|
if (ty)
|
|
|
|
return follow(*ty);
|
|
|
|
|
|
|
|
TypePackId* tp = module->astTypePacks.find(expr);
|
|
|
|
if (tp)
|
|
|
|
return flattenPack(*tp);
|
|
|
|
|
2022-09-08 22:44:50 +01:00
|
|
|
return singletonTypes->anyType;
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
|
2022-06-24 02:44:07 +01:00
|
|
|
TypeId lookupAnnotation(AstType* annotation)
|
|
|
|
{
|
|
|
|
TypeId* ty = module->astResolvedTypes.find(annotation);
|
|
|
|
LUAU_ASSERT(ty);
|
|
|
|
return follow(*ty);
|
|
|
|
}
|
|
|
|
|
2022-08-04 22:27:28 +01:00
|
|
|
TypePackId lookupPackAnnotation(AstTypePack* annotation)
|
|
|
|
{
|
|
|
|
TypePackId* tp = module->astResolvedTypePacks.find(annotation);
|
|
|
|
LUAU_ASSERT(tp);
|
|
|
|
return follow(*tp);
|
|
|
|
}
|
|
|
|
|
2022-06-24 02:44:07 +01:00
|
|
|
TypePackId reconstructPack(AstArray<AstExpr*> exprs, TypeArena& arena)
|
|
|
|
{
|
2022-07-29 04:41:13 +01:00
|
|
|
if (exprs.size == 0)
|
|
|
|
return arena.addTypePack(TypePack{{}, std::nullopt});
|
|
|
|
|
2022-06-24 02:44:07 +01:00
|
|
|
std::vector<TypeId> head;
|
|
|
|
|
|
|
|
for (size_t i = 0; i < exprs.size - 1; ++i)
|
|
|
|
{
|
|
|
|
head.push_back(lookupType(exprs.data[i]));
|
|
|
|
}
|
|
|
|
|
|
|
|
TypePackId tail = lookupPack(exprs.data[exprs.size - 1]);
|
|
|
|
return arena.addTypePack(TypePack{head, tail});
|
|
|
|
}
|
|
|
|
|
2022-07-29 04:41:13 +01:00
|
|
|
Scope* findInnermostScope(Location location)
|
2022-06-24 02:44:07 +01:00
|
|
|
{
|
2022-07-29 04:41:13 +01:00
|
|
|
Scope* bestScope = module->getModuleScope().get();
|
|
|
|
Location bestLocation = module->scopes[0].first;
|
2022-06-24 02:44:07 +01:00
|
|
|
|
2022-07-29 04:41:13 +01:00
|
|
|
for (size_t i = 0; i < module->scopes.size(); ++i)
|
2022-06-24 02:44:07 +01:00
|
|
|
{
|
2022-07-29 04:41:13 +01:00
|
|
|
auto& [scopeBounds, scope] = module->scopes[i];
|
2022-06-24 02:44:07 +01:00
|
|
|
if (scopeBounds.encloses(location))
|
|
|
|
{
|
|
|
|
if (scopeBounds.begin > bestLocation.begin || scopeBounds.end < bestLocation.end)
|
|
|
|
{
|
|
|
|
bestScope = scope.get();
|
|
|
|
bestLocation = scopeBounds;
|
|
|
|
}
|
|
|
|
}
|
2022-07-01 00:29:02 +01:00
|
|
|
else if (scopeBounds.begin > location.end)
|
2022-06-24 02:44:07 +01:00
|
|
|
{
|
|
|
|
// TODO: Is this sound? This relies on the fact that scopes are inserted
|
|
|
|
// into the scope list in the order that they appear in the AST.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return bestScope;
|
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
void visit(AstStat* stat)
|
|
|
|
{
|
|
|
|
auto pusher = pushStack(stat);
|
|
|
|
|
|
|
|
if (0)
|
2022-09-02 00:00:14 +01:00
|
|
|
{
|
|
|
|
}
|
2022-08-18 22:04:33 +01:00
|
|
|
else if (auto s = stat->as<AstStatBlock>())
|
|
|
|
return visit(s);
|
|
|
|
else if (auto s = stat->as<AstStatIf>())
|
|
|
|
return visit(s);
|
|
|
|
else if (auto s = stat->as<AstStatWhile>())
|
|
|
|
return visit(s);
|
|
|
|
else if (auto s = stat->as<AstStatRepeat>())
|
|
|
|
return visit(s);
|
|
|
|
else if (auto s = stat->as<AstStatBreak>())
|
|
|
|
return visit(s);
|
|
|
|
else if (auto s = stat->as<AstStatContinue>())
|
|
|
|
return visit(s);
|
|
|
|
else if (auto s = stat->as<AstStatReturn>())
|
|
|
|
return visit(s);
|
|
|
|
else if (auto s = stat->as<AstStatExpr>())
|
|
|
|
return visit(s);
|
|
|
|
else if (auto s = stat->as<AstStatLocal>())
|
|
|
|
return visit(s);
|
|
|
|
else if (auto s = stat->as<AstStatFor>())
|
|
|
|
return visit(s);
|
|
|
|
else if (auto s = stat->as<AstStatForIn>())
|
|
|
|
return visit(s);
|
|
|
|
else if (auto s = stat->as<AstStatAssign>())
|
|
|
|
return visit(s);
|
|
|
|
else if (auto s = stat->as<AstStatCompoundAssign>())
|
|
|
|
return visit(s);
|
|
|
|
else if (auto s = stat->as<AstStatFunction>())
|
|
|
|
return visit(s);
|
|
|
|
else if (auto s = stat->as<AstStatLocalFunction>())
|
|
|
|
return visit(s);
|
|
|
|
else if (auto s = stat->as<AstStatTypeAlias>())
|
|
|
|
return visit(s);
|
|
|
|
else if (auto s = stat->as<AstStatDeclareFunction>())
|
|
|
|
return visit(s);
|
|
|
|
else if (auto s = stat->as<AstStatDeclareGlobal>())
|
|
|
|
return visit(s);
|
|
|
|
else if (auto s = stat->as<AstStatDeclareClass>())
|
|
|
|
return visit(s);
|
|
|
|
else if (auto s = stat->as<AstStatError>())
|
|
|
|
return visit(s);
|
|
|
|
else
|
|
|
|
LUAU_ASSERT(!"TypeChecker2 encountered an unknown node type");
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstStatBlock* block)
|
|
|
|
{
|
|
|
|
auto StackPusher = pushStack(block);
|
|
|
|
|
|
|
|
for (AstStat* statement : block->body)
|
|
|
|
visit(statement);
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstStatIf* ifStatement)
|
|
|
|
{
|
|
|
|
visit(ifStatement->condition);
|
|
|
|
visit(ifStatement->thenbody);
|
|
|
|
if (ifStatement->elsebody)
|
|
|
|
visit(ifStatement->elsebody);
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstStatWhile* whileStatement)
|
|
|
|
{
|
|
|
|
visit(whileStatement->condition);
|
|
|
|
visit(whileStatement->body);
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstStatRepeat* repeatStatement)
|
|
|
|
{
|
|
|
|
visit(repeatStatement->body);
|
|
|
|
visit(repeatStatement->condition);
|
|
|
|
}
|
|
|
|
|
2022-09-02 00:00:14 +01:00
|
|
|
void visit(AstStatBreak*) {}
|
2022-08-18 22:04:33 +01:00
|
|
|
|
2022-09-02 00:00:14 +01:00
|
|
|
void visit(AstStatContinue*) {}
|
2022-08-18 22:04:33 +01:00
|
|
|
|
|
|
|
void visit(AstStatReturn* ret)
|
|
|
|
{
|
|
|
|
Scope* scope = findInnermostScope(ret->location);
|
|
|
|
TypePackId expectedRetType = scope->returnType;
|
|
|
|
|
|
|
|
TypeArena arena;
|
|
|
|
TypePackId actualRetType = reconstructPack(ret->list, arena);
|
|
|
|
|
|
|
|
UnifierSharedState sharedState{&ice};
|
2022-09-08 22:44:50 +01:00
|
|
|
Unifier u{&arena, singletonTypes, Mode::Strict, stack.back(), ret->location, Covariant, sharedState};
|
2022-08-18 22:04:33 +01:00
|
|
|
u.anyIsTop = true;
|
|
|
|
|
|
|
|
u.tryUnify(actualRetType, expectedRetType);
|
|
|
|
const bool ok = u.errors.empty() && u.log.empty();
|
|
|
|
|
|
|
|
if (!ok)
|
|
|
|
{
|
|
|
|
for (const TypeError& e : u.errors)
|
|
|
|
reportError(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (AstExpr* expr : ret->list)
|
|
|
|
visit(expr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstStatExpr* expr)
|
|
|
|
{
|
|
|
|
visit(expr->expr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstStatLocal* local)
|
2022-06-24 02:44:07 +01:00
|
|
|
{
|
|
|
|
for (size_t i = 0; i < local->values.size; ++i)
|
|
|
|
{
|
|
|
|
AstExpr* value = local->values.data[i];
|
2022-08-18 22:04:33 +01:00
|
|
|
|
|
|
|
visit(value);
|
|
|
|
|
2022-06-24 02:44:07 +01:00
|
|
|
if (i == local->values.size - 1)
|
|
|
|
{
|
|
|
|
if (i < local->values.size)
|
|
|
|
{
|
|
|
|
TypePackId valueTypes = lookupPack(value);
|
|
|
|
auto it = begin(valueTypes);
|
|
|
|
for (size_t j = i; j < local->vars.size; ++j)
|
|
|
|
{
|
|
|
|
if (it == end(valueTypes))
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
AstLocal* var = local->vars.data[i];
|
|
|
|
if (var->annotation)
|
|
|
|
{
|
|
|
|
TypeId varType = lookupAnnotation(var->annotation);
|
2022-09-08 22:44:50 +01:00
|
|
|
if (!isSubtype(*it, varType, stack.back(), singletonTypes, ice))
|
2022-06-24 02:44:07 +01:00
|
|
|
{
|
|
|
|
reportError(TypeMismatch{varType, *it}, value->location);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
TypeId valueType = lookupType(value);
|
|
|
|
AstLocal* var = local->vars.data[i];
|
|
|
|
|
|
|
|
if (var->annotation)
|
|
|
|
{
|
|
|
|
TypeId varType = lookupAnnotation(var->annotation);
|
2022-09-08 22:44:50 +01:00
|
|
|
if (!isSubtype(varType, valueType, stack.back(), singletonTypes, ice))
|
2022-06-24 02:44:07 +01:00
|
|
|
{
|
|
|
|
reportError(TypeMismatch{varType, valueType}, value->location);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-08-18 22:04:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstStatFor* forStatement)
|
|
|
|
{
|
|
|
|
if (forStatement->var->annotation)
|
|
|
|
visit(forStatement->var->annotation);
|
|
|
|
|
|
|
|
visit(forStatement->from);
|
|
|
|
visit(forStatement->to);
|
|
|
|
if (forStatement->step)
|
|
|
|
visit(forStatement->step);
|
|
|
|
visit(forStatement->body);
|
|
|
|
}
|
|
|
|
|
2022-09-02 00:00:14 +01:00
|
|
|
// "Render" a type pack out to an array of a given length. Expands
|
|
|
|
// variadics and various other things to get there.
|
2022-09-08 22:44:50 +01:00
|
|
|
std::vector<TypeId> flatten(TypeArena& arena, TypePackId pack, size_t length)
|
2022-09-02 00:00:14 +01:00
|
|
|
{
|
|
|
|
std::vector<TypeId> result;
|
|
|
|
|
|
|
|
auto it = begin(pack);
|
|
|
|
auto endIt = end(pack);
|
|
|
|
|
|
|
|
while (it != endIt)
|
|
|
|
{
|
|
|
|
result.push_back(*it);
|
|
|
|
|
|
|
|
if (result.size() >= length)
|
|
|
|
return result;
|
|
|
|
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!it.tail())
|
|
|
|
return result;
|
|
|
|
|
|
|
|
TypePackId tail = *it.tail();
|
|
|
|
if (get<TypePack>(tail))
|
|
|
|
LUAU_ASSERT(0);
|
|
|
|
else if (auto vtp = get<VariadicTypePack>(tail))
|
|
|
|
{
|
|
|
|
while (result.size() < length)
|
|
|
|
result.push_back(vtp->ty);
|
|
|
|
}
|
|
|
|
else if (get<FreeTypePack>(tail) || get<GenericTypePack>(tail))
|
|
|
|
{
|
|
|
|
while (result.size() < length)
|
|
|
|
result.push_back(arena.addType(FreeTypeVar{nullptr}));
|
|
|
|
}
|
|
|
|
else if (auto etp = get<Unifiable::Error>(tail))
|
|
|
|
{
|
|
|
|
while (result.size() < length)
|
2022-09-08 22:44:50 +01:00
|
|
|
result.push_back(singletonTypes->errorRecoveryType());
|
2022-09-02 00:00:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
void visit(AstStatForIn* forInStatement)
|
|
|
|
{
|
|
|
|
for (AstLocal* local : forInStatement->vars)
|
|
|
|
{
|
|
|
|
if (local->annotation)
|
|
|
|
visit(local->annotation);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (AstExpr* expr : forInStatement->values)
|
|
|
|
visit(expr);
|
2022-06-24 02:44:07 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
visit(forInStatement->body);
|
2022-09-02 00:00:14 +01:00
|
|
|
|
|
|
|
// Rule out crazy stuff. Maybe possible if the file is not syntactically valid.
|
|
|
|
if (!forInStatement->vars.size || !forInStatement->values.size)
|
|
|
|
return;
|
|
|
|
|
|
|
|
NotNull<Scope> scope = stack.back();
|
|
|
|
TypeArena tempArena;
|
|
|
|
|
|
|
|
std::vector<TypeId> variableTypes;
|
|
|
|
for (AstLocal* var : forInStatement->vars)
|
|
|
|
{
|
|
|
|
std::optional<TypeId> ty = scope->lookup(var);
|
|
|
|
LUAU_ASSERT(ty);
|
|
|
|
variableTypes.emplace_back(*ty);
|
|
|
|
}
|
|
|
|
|
|
|
|
// ugh. There's nothing in the AST to hang a whole type pack on for the
|
|
|
|
// set of iteratees, so we have to piece it back together by hand.
|
|
|
|
std::vector<TypeId> valueTypes;
|
|
|
|
for (size_t i = 0; i < forInStatement->values.size - 1; ++i)
|
|
|
|
valueTypes.emplace_back(lookupType(forInStatement->values.data[i]));
|
|
|
|
TypePackId iteratorTail = lookupPack(forInStatement->values.data[forInStatement->values.size - 1]);
|
|
|
|
TypePackId iteratorPack = tempArena.addTypePack(valueTypes, iteratorTail);
|
|
|
|
|
|
|
|
// ... and then expand it out to 3 values (if possible)
|
|
|
|
const std::vector<TypeId> iteratorTypes = flatten(tempArena, iteratorPack, 3);
|
|
|
|
if (iteratorTypes.empty())
|
|
|
|
{
|
|
|
|
reportError(GenericError{"for..in loops require at least one value to iterate over. Got zero"}, getLocation(forInStatement->values));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
TypeId iteratorTy = follow(iteratorTypes[0]);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If the first iterator argument is a function
|
|
|
|
* * There must be 1 to 3 iterator arguments. Name them (nextTy,
|
|
|
|
* arrayTy, startIndexTy)
|
|
|
|
* * The return type of nextTy() must correspond to the variables'
|
|
|
|
* types and counts. HOWEVER the first iterator will never be nil.
|
|
|
|
* * The first return value of nextTy must be compatible with
|
|
|
|
* startIndexTy.
|
|
|
|
* * The first argument to nextTy() must be compatible with arrayTy if
|
|
|
|
* present. nil if not.
|
|
|
|
* * The second argument to nextTy() must be compatible with
|
|
|
|
* startIndexTy if it is present. Else, it must be compatible with
|
|
|
|
* nil.
|
|
|
|
* * nextTy() must be callable with only 2 arguments.
|
|
|
|
*/
|
|
|
|
if (const FunctionTypeVar* nextFn = get<FunctionTypeVar>(iteratorTy))
|
|
|
|
{
|
|
|
|
if (iteratorTypes.size() < 1 || iteratorTypes.size() > 3)
|
|
|
|
reportError(GenericError{"for..in loops must be passed (next, [table[, state]])"}, getLocation(forInStatement->values));
|
|
|
|
|
|
|
|
// It is okay if there aren't enough iterators, but the iteratee must provide enough.
|
|
|
|
std::vector<TypeId> expectedVariableTypes = flatten(tempArena, nextFn->retTypes, variableTypes.size());
|
|
|
|
if (expectedVariableTypes.size() < variableTypes.size())
|
|
|
|
reportError(GenericError{"next() does not return enough values"}, forInStatement->vars.data[0]->location);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < std::min(expectedVariableTypes.size(), variableTypes.size()); ++i)
|
|
|
|
reportErrors(tryUnify(scope, forInStatement->vars.data[i]->location, variableTypes[i], expectedVariableTypes[i]));
|
|
|
|
|
|
|
|
// nextFn is going to be invoked with (arrayTy, startIndexTy)
|
|
|
|
|
|
|
|
// It will be passed two arguments on every iteration save the
|
|
|
|
// first.
|
|
|
|
|
|
|
|
// It may be invoked with 0 or 1 argument on the first iteration.
|
|
|
|
// This depends on the types in iterateePack and therefore
|
|
|
|
// iteratorTypes.
|
|
|
|
|
|
|
|
// If iteratorTypes is too short to be a valid call to nextFn, we have to report a count mismatch error.
|
|
|
|
// If 2 is too short to be a valid call to nextFn, we have to report a count mismatch error.
|
|
|
|
// If 2 is too long to be a valid call to nextFn, we have to report a count mismatch error.
|
|
|
|
auto [minCount, maxCount] = getParameterExtents(TxnLog::empty(), nextFn->argTypes);
|
|
|
|
|
|
|
|
if (minCount > 2)
|
|
|
|
reportError(CountMismatch{2, minCount, CountMismatch::Arg}, forInStatement->vars.data[0]->location);
|
|
|
|
if (maxCount && *maxCount < 2)
|
|
|
|
reportError(CountMismatch{2, *maxCount, CountMismatch::Arg}, forInStatement->vars.data[0]->location);
|
|
|
|
|
|
|
|
const std::vector<TypeId> flattenedArgTypes = flatten(tempArena, nextFn->argTypes, 2);
|
|
|
|
const auto [argTypes, argsTail] = Luau::flatten(nextFn->argTypes);
|
|
|
|
|
|
|
|
size_t firstIterationArgCount = iteratorTypes.empty() ? 0 : iteratorTypes.size() - 1;
|
|
|
|
size_t actualArgCount = expectedVariableTypes.size();
|
|
|
|
|
|
|
|
if (firstIterationArgCount < minCount)
|
|
|
|
reportError(CountMismatch{2, firstIterationArgCount, CountMismatch::Arg}, forInStatement->vars.data[0]->location);
|
|
|
|
else if (actualArgCount < minCount)
|
|
|
|
reportError(CountMismatch{2, actualArgCount, CountMismatch::Arg}, forInStatement->vars.data[0]->location);
|
|
|
|
|
|
|
|
if (iteratorTypes.size() >= 2 && flattenedArgTypes.size() > 0)
|
|
|
|
{
|
|
|
|
size_t valueIndex = forInStatement->values.size > 1 ? 1 : 0;
|
|
|
|
reportErrors(tryUnify(scope, forInStatement->values.data[valueIndex]->location, iteratorTypes[1], flattenedArgTypes[0]));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (iteratorTypes.size() == 3 && flattenedArgTypes.size() > 1)
|
|
|
|
{
|
|
|
|
size_t valueIndex = forInStatement->values.size > 2 ? 2 : 0;
|
|
|
|
reportErrors(tryUnify(scope, forInStatement->values.data[valueIndex]->location, iteratorTypes[2], flattenedArgTypes[1]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (const TableTypeVar* ttv = get<TableTypeVar>(iteratorTy))
|
|
|
|
{
|
|
|
|
if ((forInStatement->vars.size == 1 || forInStatement->vars.size == 2) && ttv->indexer)
|
|
|
|
{
|
|
|
|
reportErrors(tryUnify(scope, forInStatement->vars.data[0]->location, variableTypes[0], ttv->indexer->indexType));
|
|
|
|
if (variableTypes.size() == 2)
|
|
|
|
reportErrors(tryUnify(scope, forInStatement->vars.data[1]->location, variableTypes[1], ttv->indexer->indexResultType));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
reportError(GenericError{"Cannot iterate over a table without indexer"}, forInStatement->values.data[0]->location);
|
|
|
|
}
|
|
|
|
else if (get<AnyTypeVar>(iteratorTy) || get<ErrorTypeVar>(iteratorTy))
|
|
|
|
{
|
|
|
|
// nothing
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
reportError(CannotCallNonFunction{iteratorTy}, forInStatement->values.data[0]->location);
|
|
|
|
}
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
void visit(AstStatAssign* assign)
|
2022-06-17 01:54:42 +01:00
|
|
|
{
|
|
|
|
size_t count = std::min(assign->vars.size, assign->values.size);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < count; ++i)
|
|
|
|
{
|
|
|
|
AstExpr* lhs = assign->vars.data[i];
|
2022-08-18 22:04:33 +01:00
|
|
|
visit(lhs);
|
2022-07-01 00:29:02 +01:00
|
|
|
TypeId lhsType = lookupType(lhs);
|
2022-06-17 01:54:42 +01:00
|
|
|
|
|
|
|
AstExpr* rhs = assign->values.data[i];
|
2022-08-18 22:04:33 +01:00
|
|
|
visit(rhs);
|
2022-07-01 00:29:02 +01:00
|
|
|
TypeId rhsType = lookupType(rhs);
|
2022-06-17 01:54:42 +01:00
|
|
|
|
2022-09-08 22:44:50 +01:00
|
|
|
if (!isSubtype(rhsType, lhsType, stack.back(), singletonTypes, ice))
|
2022-06-17 01:54:42 +01:00
|
|
|
{
|
2022-07-01 00:29:02 +01:00
|
|
|
reportError(TypeMismatch{lhsType, rhsType}, rhs->location);
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
}
|
2022-08-18 22:04:33 +01:00
|
|
|
}
|
2022-06-17 01:54:42 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
void visit(AstStatCompoundAssign* stat)
|
|
|
|
{
|
|
|
|
visit(stat->var);
|
|
|
|
visit(stat->value);
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
void visit(AstStatFunction* stat)
|
2022-06-24 02:44:07 +01:00
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
visit(stat->name);
|
|
|
|
visit(stat->func);
|
|
|
|
}
|
2022-06-24 02:44:07 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
void visit(AstStatLocalFunction* stat)
|
|
|
|
{
|
|
|
|
visit(stat->func);
|
|
|
|
}
|
2022-06-24 02:44:07 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
void visit(const AstTypeList* typeList)
|
|
|
|
{
|
|
|
|
for (AstType* ty : typeList->types)
|
|
|
|
visit(ty);
|
2022-06-24 02:44:07 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
if (typeList->tailType)
|
|
|
|
visit(typeList->tailType);
|
|
|
|
}
|
2022-06-24 02:44:07 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
void visit(AstStatTypeAlias* stat)
|
|
|
|
{
|
|
|
|
for (const AstGenericType& el : stat->generics)
|
2022-06-24 02:44:07 +01:00
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
if (el.defaultValue)
|
|
|
|
visit(el.defaultValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const AstGenericTypePack& el : stat->genericPacks)
|
|
|
|
{
|
|
|
|
if (el.defaultValue)
|
|
|
|
visit(el.defaultValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
visit(stat->type);
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstTypeList types)
|
|
|
|
{
|
|
|
|
for (AstType* type : types.types)
|
|
|
|
visit(type);
|
|
|
|
if (types.tailType)
|
|
|
|
visit(types.tailType);
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstStatDeclareFunction* stat)
|
|
|
|
{
|
|
|
|
visit(stat->params);
|
|
|
|
visit(stat->retTypes);
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstStatDeclareGlobal* stat)
|
|
|
|
{
|
|
|
|
visit(stat->type);
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstStatDeclareClass* stat)
|
|
|
|
{
|
|
|
|
for (const AstDeclaredClassProp& prop : stat->props)
|
|
|
|
visit(prop.ty);
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstStatError* stat)
|
|
|
|
{
|
|
|
|
for (AstExpr* expr : stat->expressions)
|
|
|
|
visit(expr);
|
|
|
|
|
|
|
|
for (AstStat* s : stat->statements)
|
|
|
|
visit(s);
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstExpr* expr)
|
|
|
|
{
|
|
|
|
auto StackPusher = pushStack(expr);
|
|
|
|
|
|
|
|
if (0)
|
2022-09-02 00:00:14 +01:00
|
|
|
{
|
|
|
|
}
|
2022-08-18 22:04:33 +01:00
|
|
|
else if (auto e = expr->as<AstExprGroup>())
|
|
|
|
return visit(e);
|
|
|
|
else if (auto e = expr->as<AstExprConstantNil>())
|
|
|
|
return visit(e);
|
|
|
|
else if (auto e = expr->as<AstExprConstantBool>())
|
|
|
|
return visit(e);
|
|
|
|
else if (auto e = expr->as<AstExprConstantNumber>())
|
|
|
|
return visit(e);
|
|
|
|
else if (auto e = expr->as<AstExprConstantString>())
|
|
|
|
return visit(e);
|
|
|
|
else if (auto e = expr->as<AstExprLocal>())
|
|
|
|
return visit(e);
|
|
|
|
else if (auto e = expr->as<AstExprGlobal>())
|
|
|
|
return visit(e);
|
|
|
|
else if (auto e = expr->as<AstExprVarargs>())
|
|
|
|
return visit(e);
|
|
|
|
else if (auto e = expr->as<AstExprCall>())
|
|
|
|
return visit(e);
|
|
|
|
else if (auto e = expr->as<AstExprIndexName>())
|
|
|
|
return visit(e);
|
|
|
|
else if (auto e = expr->as<AstExprIndexExpr>())
|
|
|
|
return visit(e);
|
|
|
|
else if (auto e = expr->as<AstExprFunction>())
|
|
|
|
return visit(e);
|
|
|
|
else if (auto e = expr->as<AstExprTable>())
|
|
|
|
return visit(e);
|
|
|
|
else if (auto e = expr->as<AstExprUnary>())
|
|
|
|
return visit(e);
|
|
|
|
else if (auto e = expr->as<AstExprBinary>())
|
|
|
|
return visit(e);
|
|
|
|
else if (auto e = expr->as<AstExprTypeAssertion>())
|
|
|
|
return visit(e);
|
|
|
|
else if (auto e = expr->as<AstExprIfElse>())
|
|
|
|
return visit(e);
|
|
|
|
else if (auto e = expr->as<AstExprError>())
|
|
|
|
return visit(e);
|
|
|
|
else
|
|
|
|
LUAU_ASSERT(!"TypeChecker2 encountered an unknown expression type");
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstExprGroup* expr)
|
|
|
|
{
|
|
|
|
visit(expr->expr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstExprConstantNil* expr)
|
|
|
|
{
|
|
|
|
// TODO!
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstExprConstantBool* expr)
|
|
|
|
{
|
|
|
|
// TODO!
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstExprConstantNumber* number)
|
|
|
|
{
|
|
|
|
TypeId actualType = lookupType(number);
|
2022-09-08 22:44:50 +01:00
|
|
|
TypeId numberType = singletonTypes->numberType;
|
2022-08-18 22:04:33 +01:00
|
|
|
|
2022-09-08 22:44:50 +01:00
|
|
|
if (!isSubtype(numberType, actualType, stack.back(), singletonTypes, ice))
|
2022-08-18 22:04:33 +01:00
|
|
|
{
|
|
|
|
reportError(TypeMismatch{actualType, numberType}, number->location);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstExprConstantString* string)
|
|
|
|
{
|
|
|
|
TypeId actualType = lookupType(string);
|
2022-09-08 22:44:50 +01:00
|
|
|
TypeId stringType = singletonTypes->stringType;
|
2022-08-18 22:04:33 +01:00
|
|
|
|
2022-09-08 22:44:50 +01:00
|
|
|
if (!isSubtype(stringType, actualType, stack.back(), singletonTypes, ice))
|
2022-08-18 22:04:33 +01:00
|
|
|
{
|
|
|
|
reportError(TypeMismatch{actualType, stringType}, string->location);
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
2022-08-18 22:04:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstExprLocal* expr)
|
|
|
|
{
|
|
|
|
// TODO!
|
|
|
|
}
|
2022-06-24 02:44:07 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
void visit(AstExprGlobal* expr)
|
|
|
|
{
|
|
|
|
// TODO!
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstExprVarargs* expr)
|
|
|
|
{
|
|
|
|
// TODO!
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
void visit(AstExprCall* call)
|
2022-06-17 01:54:42 +01:00
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
visit(call->func);
|
|
|
|
|
|
|
|
for (AstExpr* arg : call->args)
|
|
|
|
visit(arg);
|
|
|
|
|
2022-07-01 00:29:02 +01:00
|
|
|
TypeArena arena;
|
|
|
|
Instantiation instantiation{TxnLog::empty(), &arena, TypeLevel{}};
|
|
|
|
|
2022-06-17 01:54:42 +01:00
|
|
|
TypePackId expectedRetType = lookupPack(call);
|
|
|
|
TypeId functionType = lookupType(call->func);
|
2022-07-01 00:29:02 +01:00
|
|
|
LUAU_ASSERT(functionType);
|
2022-06-17 01:54:42 +01:00
|
|
|
|
2022-09-02 00:00:14 +01:00
|
|
|
if (get<AnyTypeVar>(functionType) || get<ErrorTypeVar>(functionType))
|
|
|
|
return;
|
|
|
|
|
|
|
|
// TODO: Lots of other types are callable: intersections of functions
|
|
|
|
// and things with the __call metamethod.
|
|
|
|
if (!get<FunctionTypeVar>(functionType))
|
|
|
|
{
|
|
|
|
reportError(CannotCallNonFunction{functionType}, call->func->location);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
TypeId instantiatedFunctionType = follow(instantiation.substitute(functionType).value_or(nullptr));
|
|
|
|
|
2022-06-17 01:54:42 +01:00
|
|
|
TypePack args;
|
2022-08-18 22:04:33 +01:00
|
|
|
for (AstExpr* arg : call->args)
|
2022-06-17 01:54:42 +01:00
|
|
|
{
|
|
|
|
TypeId argTy = module->astTypes[arg];
|
|
|
|
LUAU_ASSERT(argTy);
|
|
|
|
args.head.push_back(argTy);
|
|
|
|
}
|
|
|
|
|
|
|
|
TypePackId argsTp = arena.addTypePack(args);
|
|
|
|
FunctionTypeVar ftv{argsTp, expectedRetType};
|
|
|
|
TypeId expectedType = arena.addType(ftv);
|
2022-09-02 00:00:14 +01:00
|
|
|
|
2022-09-08 22:44:50 +01:00
|
|
|
if (!isSubtype(expectedType, instantiatedFunctionType, stack.back(), singletonTypes, ice))
|
2022-06-17 01:54:42 +01:00
|
|
|
{
|
|
|
|
CloneState cloneState;
|
2022-09-02 00:00:14 +01:00
|
|
|
expectedType = clone(expectedType, module->internalTypes, cloneState);
|
2022-06-17 01:54:42 +01:00
|
|
|
reportError(TypeMismatch{expectedType, functionType}, call->location);
|
|
|
|
}
|
2022-08-18 22:04:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstExprIndexName* indexName)
|
|
|
|
{
|
|
|
|
TypeId leftType = lookupType(indexName->expr);
|
|
|
|
TypeId resultType = lookupType(indexName);
|
|
|
|
|
|
|
|
// leftType must have a property called indexName->index
|
|
|
|
|
2022-09-02 00:00:14 +01:00
|
|
|
std::optional<TypeId> ty =
|
|
|
|
getIndexTypeFromType(module->getModuleScope(), leftType, indexName->index.value, indexName->location, /* addErrors */ true);
|
2022-08-18 22:04:33 +01:00
|
|
|
if (ty)
|
|
|
|
{
|
2022-09-08 22:44:50 +01:00
|
|
|
if (!isSubtype(resultType, *ty, stack.back(), singletonTypes, ice))
|
2022-08-18 22:04:33 +01:00
|
|
|
{
|
|
|
|
reportError(TypeMismatch{resultType, *ty}, indexName->location);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-06-17 01:54:42 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
void visit(AstExprIndexExpr* indexExpr)
|
|
|
|
{
|
|
|
|
// TODO!
|
|
|
|
visit(indexExpr->expr);
|
|
|
|
visit(indexExpr->index);
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
void visit(AstExprFunction* fn)
|
2022-06-24 02:44:07 +01:00
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
auto StackPusher = pushStack(fn);
|
|
|
|
|
2022-06-24 02:44:07 +01:00
|
|
|
TypeId inferredFnTy = lookupType(fn);
|
|
|
|
const FunctionTypeVar* inferredFtv = get<FunctionTypeVar>(inferredFnTy);
|
|
|
|
LUAU_ASSERT(inferredFtv);
|
|
|
|
|
|
|
|
auto argIt = begin(inferredFtv->argTypes);
|
|
|
|
for (const auto& arg : fn->args)
|
|
|
|
{
|
|
|
|
if (argIt == end(inferredFtv->argTypes))
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (arg->annotation)
|
|
|
|
{
|
|
|
|
TypeId inferredArgTy = *argIt;
|
|
|
|
TypeId annotatedArgTy = lookupAnnotation(arg->annotation);
|
|
|
|
|
2022-09-08 22:44:50 +01:00
|
|
|
if (!isSubtype(annotatedArgTy, inferredArgTy, stack.back(), singletonTypes, ice))
|
2022-06-24 02:44:07 +01:00
|
|
|
{
|
|
|
|
reportError(TypeMismatch{annotatedArgTy, inferredArgTy}, arg->location);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
++argIt;
|
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
visit(fn->body);
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
void visit(AstExprTable* expr)
|
2022-06-17 01:54:42 +01:00
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
// TODO!
|
|
|
|
for (const AstExprTable::Item& item : expr->items)
|
2022-06-17 01:54:42 +01:00
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
if (item.key)
|
|
|
|
visit(item.key);
|
|
|
|
visit(item.value);
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
void visit(AstExprUnary* expr)
|
2022-06-17 01:54:42 +01:00
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
// TODO!
|
|
|
|
visit(expr->expr);
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
void visit(AstExprBinary* expr)
|
2022-06-17 01:54:42 +01:00
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
// TODO!
|
|
|
|
visit(expr->left);
|
|
|
|
visit(expr->right);
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
void visit(AstExprTypeAssertion* expr)
|
2022-08-11 21:42:54 +01:00
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
visit(expr->expr);
|
|
|
|
visit(expr->annotation);
|
|
|
|
|
2022-08-11 21:42:54 +01:00
|
|
|
TypeId annotationType = lookupAnnotation(expr->annotation);
|
|
|
|
TypeId computedType = lookupType(expr->expr);
|
|
|
|
|
|
|
|
// Note: As an optimization, we try 'number <: number | string' first, as that is the more likely case.
|
2022-09-08 22:44:50 +01:00
|
|
|
if (isSubtype(annotationType, computedType, stack.back(), singletonTypes, ice))
|
2022-08-18 22:04:33 +01:00
|
|
|
return;
|
2022-08-11 21:42:54 +01:00
|
|
|
|
2022-09-08 22:44:50 +01:00
|
|
|
if (isSubtype(computedType, annotationType, stack.back(), singletonTypes, ice))
|
2022-08-18 22:04:33 +01:00
|
|
|
return;
|
2022-08-11 21:42:54 +01:00
|
|
|
|
|
|
|
reportError(TypesAreUnrelated{computedType, annotationType}, expr->location);
|
2022-08-18 22:04:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstExprIfElse* expr)
|
|
|
|
{
|
|
|
|
// TODO!
|
|
|
|
visit(expr->condition);
|
|
|
|
visit(expr->trueExpr);
|
|
|
|
visit(expr->falseExpr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstExprError* expr)
|
|
|
|
{
|
|
|
|
// TODO!
|
|
|
|
for (AstExpr* e : expr->expressions)
|
|
|
|
visit(e);
|
2022-08-11 21:42:54 +01:00
|
|
|
}
|
|
|
|
|
2022-07-01 00:29:02 +01:00
|
|
|
/** Extract a TypeId for the first type of the provided pack.
|
|
|
|
*
|
|
|
|
* Note that this may require modifying some types. I hope this doesn't cause problems!
|
|
|
|
*/
|
|
|
|
TypeId flattenPack(TypePackId pack)
|
|
|
|
{
|
|
|
|
pack = follow(pack);
|
|
|
|
|
2022-07-14 23:39:35 +01:00
|
|
|
while (true)
|
2022-07-01 00:29:02 +01:00
|
|
|
{
|
2022-07-14 23:39:35 +01:00
|
|
|
auto tp = get<TypePack>(pack);
|
|
|
|
if (tp && tp->head.empty() && tp->tail)
|
2022-07-01 00:29:02 +01:00
|
|
|
pack = *tp->tail;
|
2022-07-14 23:39:35 +01:00
|
|
|
else
|
|
|
|
break;
|
2022-07-01 00:29:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (auto ty = first(pack))
|
|
|
|
return *ty;
|
|
|
|
else if (auto vtp = get<VariadicTypePack>(pack))
|
|
|
|
return vtp->ty;
|
|
|
|
else if (auto ftp = get<FreeTypePack>(pack))
|
|
|
|
{
|
|
|
|
TypeId result = module->internalTypes.addType(FreeTypeVar{ftp->scope});
|
|
|
|
TypePackId freeTail = module->internalTypes.addTypePack(FreeTypePack{ftp->scope});
|
|
|
|
|
|
|
|
TypePack& resultPack = asMutable(pack)->ty.emplace<TypePack>();
|
|
|
|
resultPack.head.assign(1, result);
|
|
|
|
resultPack.tail = freeTail;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
else if (get<Unifiable::Error>(pack))
|
2022-09-08 22:44:50 +01:00
|
|
|
return singletonTypes->errorRecoveryType();
|
2022-07-01 00:29:02 +01:00
|
|
|
else
|
|
|
|
ice.ice("flattenPack got a weird pack!");
|
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
void visit(AstType* ty)
|
2022-06-24 02:44:07 +01:00
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
if (auto t = ty->as<AstTypeReference>())
|
|
|
|
return visit(t);
|
|
|
|
else if (auto t = ty->as<AstTypeTable>())
|
|
|
|
return visit(t);
|
|
|
|
else if (auto t = ty->as<AstTypeFunction>())
|
|
|
|
return visit(t);
|
|
|
|
else if (auto t = ty->as<AstTypeTypeof>())
|
|
|
|
return visit(t);
|
|
|
|
else if (auto t = ty->as<AstTypeUnion>())
|
|
|
|
return visit(t);
|
|
|
|
else if (auto t = ty->as<AstTypeIntersection>())
|
|
|
|
return visit(t);
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
void visit(AstTypeReference* ty)
|
2022-06-24 02:44:07 +01:00
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
for (const AstTypeOrPack& param : ty->parameters)
|
|
|
|
{
|
|
|
|
if (param.type)
|
|
|
|
visit(param.type);
|
|
|
|
else
|
|
|
|
visit(param.typePack);
|
|
|
|
}
|
|
|
|
|
2022-07-29 04:41:13 +01:00
|
|
|
Scope* scope = findInnermostScope(ty->location);
|
2022-08-04 22:27:28 +01:00
|
|
|
LUAU_ASSERT(scope);
|
2022-06-24 02:44:07 +01:00
|
|
|
|
|
|
|
// TODO: Imported types
|
2022-08-04 22:27:28 +01:00
|
|
|
|
2022-08-11 21:42:54 +01:00
|
|
|
std::optional<TypeFun> alias = scope->lookupType(ty->name.value);
|
2022-08-04 22:27:28 +01:00
|
|
|
|
|
|
|
if (alias.has_value())
|
|
|
|
{
|
|
|
|
size_t typesRequired = alias->typeParams.size();
|
|
|
|
size_t packsRequired = alias->typePackParams.size();
|
|
|
|
|
|
|
|
bool hasDefaultTypes = std::any_of(alias->typeParams.begin(), alias->typeParams.end(), [](auto&& el) {
|
|
|
|
return el.defaultValue.has_value();
|
|
|
|
});
|
|
|
|
|
|
|
|
bool hasDefaultPacks = std::any_of(alias->typePackParams.begin(), alias->typePackParams.end(), [](auto&& el) {
|
|
|
|
return el.defaultValue.has_value();
|
|
|
|
});
|
|
|
|
|
|
|
|
if (!ty->hasParameterList)
|
|
|
|
{
|
|
|
|
if ((!alias->typeParams.empty() && !hasDefaultTypes) || (!alias->typePackParams.empty() && !hasDefaultPacks))
|
|
|
|
{
|
|
|
|
reportError(GenericError{"Type parameter list is required"}, ty->location);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t typesProvided = 0;
|
|
|
|
size_t extraTypes = 0;
|
|
|
|
size_t packsProvided = 0;
|
|
|
|
|
|
|
|
for (const AstTypeOrPack& p : ty->parameters)
|
|
|
|
{
|
|
|
|
if (p.type)
|
|
|
|
{
|
|
|
|
if (packsProvided != 0)
|
|
|
|
{
|
|
|
|
reportError(GenericError{"Type parameters must come before type pack parameters"}, ty->location);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typesProvided < typesRequired)
|
|
|
|
{
|
|
|
|
typesProvided += 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
extraTypes += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (p.typePack)
|
|
|
|
{
|
|
|
|
TypePackId tp = lookupPackAnnotation(p.typePack);
|
|
|
|
|
|
|
|
if (typesProvided < typesRequired && size(tp) == 1 && finite(tp) && first(tp))
|
|
|
|
{
|
|
|
|
typesProvided += 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
packsProvided += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (extraTypes != 0 && packsProvided == 0)
|
|
|
|
{
|
|
|
|
packsProvided += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i = typesProvided; i < typesRequired; ++i)
|
|
|
|
{
|
|
|
|
if (alias->typeParams[i].defaultValue)
|
|
|
|
{
|
|
|
|
typesProvided += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i = packsProvided; i < packsProvided; ++i)
|
|
|
|
{
|
|
|
|
if (alias->typePackParams[i].defaultValue)
|
|
|
|
{
|
|
|
|
packsProvided += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (extraTypes == 0 && packsProvided + 1 == packsRequired)
|
|
|
|
{
|
|
|
|
packsProvided += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typesProvided != typesRequired || packsProvided != packsRequired)
|
|
|
|
{
|
|
|
|
reportError(IncorrectGenericParameterCount{
|
|
|
|
/* name */ ty->name.value,
|
|
|
|
/* typeFun */ *alias,
|
|
|
|
/* actualParameters */ typesProvided,
|
|
|
|
/* actualPackParameters */ packsProvided,
|
|
|
|
},
|
|
|
|
ty->location);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-08-11 21:42:54 +01:00
|
|
|
if (scope->lookupPack(ty->name.value))
|
2022-08-04 22:27:28 +01:00
|
|
|
{
|
|
|
|
reportError(
|
|
|
|
SwappedGenericTypeParameter{
|
|
|
|
ty->name.value,
|
|
|
|
SwappedGenericTypeParameter::Kind::Type,
|
|
|
|
},
|
|
|
|
ty->location);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
reportError(UnknownSymbol{ty->name.value, UnknownSymbol::Context::Type}, ty->location);
|
|
|
|
}
|
|
|
|
}
|
2022-08-18 22:04:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstTypeTable* table)
|
|
|
|
{
|
|
|
|
// TODO!
|
|
|
|
|
|
|
|
for (const AstTableProp& prop : table->props)
|
|
|
|
visit(prop.type);
|
|
|
|
|
|
|
|
if (table->indexer)
|
|
|
|
{
|
|
|
|
visit(table->indexer->indexType);
|
|
|
|
visit(table->indexer->resultType);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstTypeFunction* ty)
|
|
|
|
{
|
|
|
|
// TODO!
|
|
|
|
|
|
|
|
visit(ty->argTypes);
|
|
|
|
visit(ty->returnTypes);
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstTypeTypeof* ty)
|
|
|
|
{
|
|
|
|
visit(ty->expr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstTypeUnion* ty)
|
|
|
|
{
|
|
|
|
// TODO!
|
|
|
|
for (AstType* type : ty->types)
|
|
|
|
visit(type);
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstTypeIntersection* ty)
|
|
|
|
{
|
|
|
|
// TODO!
|
|
|
|
for (AstType* type : ty->types)
|
|
|
|
visit(type);
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstTypePack* pack)
|
|
|
|
{
|
|
|
|
if (auto p = pack->as<AstTypePackExplicit>())
|
|
|
|
return visit(p);
|
|
|
|
else if (auto p = pack->as<AstTypePackVariadic>())
|
|
|
|
return visit(p);
|
|
|
|
else if (auto p = pack->as<AstTypePackGeneric>())
|
|
|
|
return visit(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
void visit(AstTypePackExplicit* tp)
|
|
|
|
{
|
|
|
|
// TODO!
|
|
|
|
for (AstType* type : tp->typeList.types)
|
|
|
|
visit(type);
|
2022-08-04 22:27:28 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
if (tp->typeList.tailType)
|
|
|
|
visit(tp->typeList.tailType);
|
2022-08-04 22:27:28 +01:00
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
void visit(AstTypePackVariadic* tp)
|
2022-08-04 22:27:28 +01:00
|
|
|
{
|
2022-08-18 22:04:33 +01:00
|
|
|
// TODO!
|
|
|
|
visit(tp->variadicType);
|
2022-08-04 22:27:28 +01:00
|
|
|
}
|
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
void visit(AstTypePackGeneric* tp)
|
2022-08-04 22:27:28 +01:00
|
|
|
{
|
|
|
|
Scope* scope = findInnermostScope(tp->location);
|
|
|
|
LUAU_ASSERT(scope);
|
|
|
|
|
2022-08-11 21:42:54 +01:00
|
|
|
std::optional<TypePackId> alias = scope->lookupPack(tp->genericName.value);
|
2022-08-04 22:27:28 +01:00
|
|
|
if (!alias.has_value())
|
2022-06-24 02:44:07 +01:00
|
|
|
{
|
2022-08-11 21:42:54 +01:00
|
|
|
if (scope->lookupType(tp->genericName.value))
|
2022-08-04 22:27:28 +01:00
|
|
|
{
|
|
|
|
reportError(
|
|
|
|
SwappedGenericTypeParameter{
|
|
|
|
tp->genericName.value,
|
|
|
|
SwappedGenericTypeParameter::Kind::Pack,
|
|
|
|
},
|
|
|
|
tp->location);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
reportError(UnknownSymbol{tp->genericName.value, UnknownSymbol::Context::Type}, tp->location);
|
|
|
|
}
|
2022-06-24 02:44:07 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-02 00:00:14 +01:00
|
|
|
template<typename TID>
|
|
|
|
ErrorVec tryUnify(NotNull<Scope> scope, const Location& location, TID subTy, TID superTy)
|
|
|
|
{
|
|
|
|
UnifierSharedState sharedState{&ice};
|
2022-09-08 22:44:50 +01:00
|
|
|
Unifier u{&module->internalTypes, singletonTypes, Mode::Strict, scope, location, Covariant, sharedState};
|
2022-09-02 00:00:14 +01:00
|
|
|
u.anyIsTop = true;
|
|
|
|
u.tryUnify(subTy, superTy);
|
|
|
|
|
|
|
|
return std::move(u.errors);
|
|
|
|
}
|
|
|
|
|
|
|
|
void reportError(TypeErrorData data, const Location& location)
|
2022-06-17 01:54:42 +01:00
|
|
|
{
|
|
|
|
module->errors.emplace_back(location, sourceModule->name, std::move(data));
|
2022-09-08 22:44:50 +01:00
|
|
|
|
|
|
|
if (FFlag::DebugLuauLogSolverToJson)
|
|
|
|
logger->captureTypeCheckError(module->errors.back());
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
2022-07-01 00:29:02 +01:00
|
|
|
|
|
|
|
void reportError(TypeError e)
|
|
|
|
{
|
2022-09-02 00:00:14 +01:00
|
|
|
reportError(std::move(e.data), e.location);
|
|
|
|
}
|
|
|
|
|
|
|
|
void reportErrors(ErrorVec errors)
|
|
|
|
{
|
|
|
|
for (TypeError e : errors)
|
|
|
|
reportError(std::move(e));
|
2022-07-01 00:29:02 +01:00
|
|
|
}
|
2022-08-11 21:42:54 +01:00
|
|
|
|
2022-09-02 00:00:14 +01:00
|
|
|
std::optional<TypeId> getIndexTypeFromType(const ScopePtr& scope, TypeId type, const std::string& prop, const Location& location, bool addErrors)
|
2022-08-11 21:42:54 +01:00
|
|
|
{
|
2022-09-08 22:44:50 +01:00
|
|
|
return Luau::getIndexTypeFromType(scope, module->errors, &module->internalTypes, singletonTypes, type, prop, location, addErrors, ice);
|
2022-08-11 21:42:54 +01:00
|
|
|
}
|
2022-06-17 01:54:42 +01:00
|
|
|
};
|
|
|
|
|
2022-09-08 22:44:50 +01:00
|
|
|
void check(NotNull<SingletonTypes> singletonTypes, DcrLogger* logger, const SourceModule& sourceModule, Module* module)
|
2022-06-17 01:54:42 +01:00
|
|
|
{
|
2022-09-08 22:44:50 +01:00
|
|
|
TypeChecker2 typeChecker{singletonTypes, logger, &sourceModule, module};
|
2022-06-17 01:54:42 +01:00
|
|
|
|
2022-08-18 22:04:33 +01:00
|
|
|
typeChecker.visit(sourceModule.root);
|
2022-06-17 01:54:42 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace Luau
|