luau/Analysis/src/Module.cpp

323 lines
9.3 KiB
C++
Raw Normal View History

// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
#include "Luau/Module.h"
2022-04-07 21:53:47 +01:00
#include "Luau/Clone.h"
2022-04-14 22:57:15 +01:00
#include "Luau/Common.h"
2022-06-03 21:32:20 +01:00
#include "Luau/ConstraintGraphBuilder.h"
2022-04-14 22:57:15 +01:00
#include "Luau/Normalize.h"
2021-12-02 23:20:08 +00:00
#include "Luau/RecursionCounter.h"
#include "Luau/Scope.h"
#include "Luau/TypeInfer.h"
#include "Luau/TypePack.h"
#include "Luau/TypeVar.h"
#include "Luau/VisitTypeVar.h"
#include <algorithm>
2022-05-20 00:46:52 +01:00
LUAU_FASTFLAG(LuauLowerBoundsCalculation);
2022-06-03 21:32:20 +01:00
LUAU_FASTFLAG(DebugLuauDeferredConstraintResolution);
2022-07-21 21:36:41 +01:00
LUAU_FASTFLAGVARIABLE(LuauForceExportSurfacesToBeNormal, false);
2022-08-11 21:42:54 +01:00
LUAU_FASTFLAGVARIABLE(LuauClonePublicInterfaceLess, false);
LUAU_FASTFLAG(LuauSubstitutionReentrant);
LUAU_FASTFLAG(LuauClassTypeVarsInSubstitution);
LUAU_FASTFLAG(LuauSubstitutionFixMissingFields);
2022-01-27 21:29:34 +00:00
namespace Luau
{
static bool contains(Position pos, Comment comment)
{
if (comment.location.contains(pos))
return true;
else if (comment.type == Lexeme::BrokenComment &&
comment.location.begin <= pos) // Broken comments are broken specifically because they don't have an end
return true;
else if (comment.type == Lexeme::Comment && comment.location.end == pos)
return true;
else
return false;
}
bool isWithinComment(const SourceModule& sourceModule, Position pos)
{
auto iter = std::lower_bound(sourceModule.commentLocations.begin(), sourceModule.commentLocations.end(),
Comment{Lexeme::Comment, Location{pos, pos}}, [](const Comment& a, const Comment& b) {
return a.location.end < b.location.end;
});
if (iter == sourceModule.commentLocations.end())
return false;
if (contains(pos, *iter))
return true;
// Due to the nature of std::lower_bound, it is possible that iter points at a comment that ends
// at pos. We'll try the next comment, if it exists.
++iter;
if (iter == sourceModule.commentLocations.end())
return false;
return contains(pos, *iter);
}
2022-05-20 00:46:52 +01:00
struct ForceNormal : TypeVarOnceVisitor
{
2022-05-26 21:33:48 +01:00
const TypeArena* typeArena = nullptr;
ForceNormal(const TypeArena* typeArena)
: typeArena(typeArena)
{
}
2022-05-20 00:46:52 +01:00
bool visit(TypeId ty) override
{
2022-05-26 21:33:48 +01:00
if (ty->owningArena != typeArena)
return false;
2022-05-20 00:46:52 +01:00
asMutable(ty)->normal = true;
return true;
}
2022-05-20 00:46:52 +01:00
bool visit(TypeId ty, const FreeTypeVar& ftv) override
{
visit(ty);
return true;
}
2022-05-20 00:46:52 +01:00
bool visit(TypePackId tp, const FreeTypePack& ftp) override
{
return true;
}
};
2022-08-11 21:42:54 +01:00
struct ClonePublicInterface : Substitution
{
NotNull<Module> module;
ClonePublicInterface(const TxnLog* log, Module* module)
: Substitution(log, &module->interfaceTypes)
, module(module)
{
LUAU_ASSERT(module);
}
bool isDirty(TypeId ty) override
{
if (ty->owningArena == &module->internalTypes)
return true;
if (const FunctionTypeVar* ftv = get<FunctionTypeVar>(ty))
return ftv->level.level != 0;
if (const TableTypeVar* ttv = get<TableTypeVar>(ty))
return ttv->level.level != 0;
return false;
}
bool isDirty(TypePackId tp) override
{
return tp->owningArena == &module->internalTypes;
}
TypeId clean(TypeId ty) override
{
TypeId result = clone(ty);
if (FunctionTypeVar* ftv = getMutable<FunctionTypeVar>(result))
ftv->level = TypeLevel{0, 0};
else if (TableTypeVar* ttv = getMutable<TableTypeVar>(result))
ttv->level = TypeLevel{0, 0};
return result;
}
TypePackId clean(TypePackId tp) override
{
return clone(tp);
}
TypeId cloneType(TypeId ty)
{
LUAU_ASSERT(FFlag::LuauSubstitutionReentrant && FFlag::LuauSubstitutionFixMissingFields);
std::optional<TypeId> result = substitute(ty);
if (result)
{
return *result;
}
else
{
module->errors.push_back(TypeError{module->scopes[0].first, UnificationTooComplex{}});
return getSingletonTypes().errorRecoveryType();
}
}
TypePackId cloneTypePack(TypePackId tp)
{
LUAU_ASSERT(FFlag::LuauSubstitutionReentrant && FFlag::LuauSubstitutionFixMissingFields);
std::optional<TypePackId> result = substitute(tp);
if (result)
{
return *result;
}
else
{
module->errors.push_back(TypeError{module->scopes[0].first, UnificationTooComplex{}});
return getSingletonTypes().errorRecoveryTypePack();
}
}
TypeFun cloneTypeFun(const TypeFun& tf)
{
LUAU_ASSERT(FFlag::LuauSubstitutionReentrant && FFlag::LuauSubstitutionFixMissingFields);
std::vector<GenericTypeDefinition> typeParams;
std::vector<GenericTypePackDefinition> typePackParams;
for (GenericTypeDefinition typeParam : tf.typeParams)
{
TypeId ty = cloneType(typeParam.ty);
std::optional<TypeId> defaultValue;
if (typeParam.defaultValue)
defaultValue = cloneType(*typeParam.defaultValue);
typeParams.push_back(GenericTypeDefinition{ty, defaultValue});
}
for (GenericTypePackDefinition typePackParam : tf.typePackParams)
{
TypePackId tp = cloneTypePack(typePackParam.tp);
std::optional<TypePackId> defaultValue;
if (typePackParam.defaultValue)
defaultValue = cloneTypePack(*typePackParam.defaultValue);
typePackParams.push_back(GenericTypePackDefinition{tp, defaultValue});
}
TypeId type = cloneType(tf.type);
return TypeFun{typeParams, typePackParams, type};
}
};
Module::~Module()
{
unfreeze(interfaceTypes);
unfreeze(internalTypes);
}
2022-05-20 00:46:52 +01:00
void Module::clonePublicInterface(InternalErrorReporter& ice)
{
LUAU_ASSERT(interfaceTypes.typeVars.empty());
LUAU_ASSERT(interfaceTypes.typePacks.empty());
2021-12-02 23:20:08 +00:00
CloneState cloneState;
2022-07-29 04:41:13 +01:00
ScopePtr moduleScope = getModuleScope();
2022-07-29 04:41:13 +01:00
TypePackId returnType = moduleScope->returnType;
2022-06-03 21:32:20 +01:00
std::optional<TypePackId> varargPack = FFlag::DebugLuauDeferredConstraintResolution ? std::nullopt : moduleScope->varargPack;
std::unordered_map<Name, TypeFun>* exportedTypeBindings =
FFlag::DebugLuauDeferredConstraintResolution ? nullptr : &moduleScope->exportedTypeBindings;
2022-08-11 21:42:54 +01:00
TxnLog log;
ClonePublicInterface clonePublicInterface{&log, this};
if (FFlag::LuauClonePublicInterfaceLess)
returnType = clonePublicInterface.cloneTypePack(returnType);
else
returnType = clone(returnType, interfaceTypes, cloneState);
2022-06-03 21:32:20 +01:00
2022-07-29 04:41:13 +01:00
moduleScope->returnType = returnType;
if (varargPack)
2022-06-03 21:32:20 +01:00
{
2022-08-11 21:42:54 +01:00
if (FFlag::LuauClonePublicInterfaceLess)
varargPack = clonePublicInterface.cloneTypePack(*varargPack);
else
varargPack = clone(*varargPack, interfaceTypes, cloneState);
2022-07-29 04:41:13 +01:00
moduleScope->varargPack = varargPack;
2022-06-03 21:32:20 +01:00
}
2022-04-14 22:57:15 +01:00
2022-07-21 21:36:41 +01:00
ForceNormal forceNormal{&interfaceTypes};
2022-04-14 22:57:15 +01:00
if (FFlag::LuauLowerBoundsCalculation)
{
2022-08-18 22:04:33 +01:00
normalize(returnType, NotNull{this}, ice);
2022-07-21 21:36:41 +01:00
if (FFlag::LuauForceExportSurfacesToBeNormal)
forceNormal.traverse(returnType);
2022-06-03 21:32:20 +01:00
if (varargPack)
2022-07-21 21:36:41 +01:00
{
2022-08-18 22:04:33 +01:00
normalize(*varargPack, NotNull{this}, ice);
2022-07-21 21:36:41 +01:00
if (FFlag::LuauForceExportSurfacesToBeNormal)
forceNormal.traverse(*varargPack);
}
2022-04-14 22:57:15 +01:00
}
2022-06-03 21:32:20 +01:00
if (exportedTypeBindings)
2022-04-14 22:57:15 +01:00
{
2022-06-03 21:32:20 +01:00
for (auto& [name, tf] : *exportedTypeBindings)
2022-05-20 00:46:52 +01:00
{
2022-08-11 21:42:54 +01:00
if (FFlag::LuauClonePublicInterfaceLess)
tf = clonePublicInterface.cloneTypeFun(tf);
else
tf = clone(tf, interfaceTypes, cloneState);
2022-06-03 21:32:20 +01:00
if (FFlag::LuauLowerBoundsCalculation)
2022-05-20 00:46:52 +01:00
{
2022-08-18 22:04:33 +01:00
normalize(tf.type, NotNull{this}, ice);
2022-06-03 21:32:20 +01:00
2022-08-04 22:27:28 +01:00
// We're about to freeze the memory. We know that the flag is conservative by design. Cyclic tables
// won't be marked normal. If the types aren't normal by now, they never will be.
forceNormal.traverse(tf.type);
for (GenericTypeDefinition param : tf.typeParams)
2022-06-03 21:32:20 +01:00
{
2022-08-04 22:27:28 +01:00
forceNormal.traverse(param.ty);
2022-07-21 21:36:41 +01:00
2022-08-04 22:27:28 +01:00
if (param.defaultValue)
{
2022-08-18 22:04:33 +01:00
normalize(*param.defaultValue, NotNull{this}, ice);
2022-08-04 22:27:28 +01:00
forceNormal.traverse(*param.defaultValue);
2022-07-21 21:36:41 +01:00
}
2022-06-03 21:32:20 +01:00
}
2022-05-20 00:46:52 +01:00
}
}
2022-04-14 22:57:15 +01:00
}
2022-06-03 21:32:20 +01:00
for (TypeId ty : returnType)
2022-04-14 22:57:15 +01:00
{
if (get<GenericTypeVar>(follow(ty)))
2022-04-14 22:57:15 +01:00
{
auto t = asMutable(ty);
t->ty = AnyTypeVar{};
t->normal = true;
}
}
2022-04-21 22:04:22 +01:00
for (auto& [name, ty] : declaredGlobals)
2022-03-18 00:06:25 +00:00
{
2022-08-11 21:42:54 +01:00
if (FFlag::LuauClonePublicInterfaceLess)
ty = clonePublicInterface.cloneType(ty);
else
ty = clone(ty, interfaceTypes, cloneState);
2022-04-21 22:04:22 +01:00
if (FFlag::LuauLowerBoundsCalculation)
2022-07-21 21:36:41 +01:00
{
2022-08-18 22:04:33 +01:00
normalize(ty, NotNull{this}, ice);
2022-07-21 21:36:41 +01:00
if (FFlag::LuauForceExportSurfacesToBeNormal)
forceNormal.traverse(ty);
}
2022-03-18 00:06:25 +00:00
}
freeze(internalTypes);
freeze(interfaceTypes);
2022-05-20 00:46:52 +01:00
}
2022-05-20 00:46:52 +01:00
ScopePtr Module::getModuleScope() const
{
LUAU_ASSERT(!scopes.empty());
return scopes.front().second;
}
} // namespace Luau