2021-10-29 21:25:12 +01:00
|
|
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
|
|
#include "Luau/Unifier.h"
|
|
|
|
|
|
|
|
#include "Luau/Common.h"
|
|
|
|
#include "Luau/RecursionCounter.h"
|
2021-11-05 02:34:35 +00:00
|
|
|
#include "Luau/Scope.h"
|
2021-10-29 21:25:12 +01:00
|
|
|
#include "Luau/TypePack.h"
|
|
|
|
#include "Luau/TypeUtils.h"
|
2021-11-05 02:34:35 +00:00
|
|
|
#include "Luau/TimeTrace.h"
|
2021-11-05 15:47:21 +00:00
|
|
|
#include "Luau/VisitTypeVar.h"
|
2022-02-04 16:45:57 +00:00
|
|
|
#include "Luau/ToString.h"
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
|
|
|
|
LUAU_FASTINT(LuauTypeInferRecursionLimit);
|
|
|
|
LUAU_FASTINT(LuauTypeInferTypePackLoopLimit);
|
2022-04-15 00:57:43 +01:00
|
|
|
LUAU_FASTINT(LuauTypeInferIterationLimit);
|
|
|
|
LUAU_FASTFLAG(LuauAutocompleteDynamicLimits)
|
|
|
|
LUAU_FASTINTVARIABLE(LuauTypeInferLowerBoundsIterationLimit, 2000);
|
2021-12-10 22:05:05 +00:00
|
|
|
LUAU_FASTFLAGVARIABLE(LuauTableSubtypingVariance2, false);
|
2022-04-15 00:57:43 +01:00
|
|
|
LUAU_FASTFLAG(LuauLowerBoundsCalculation);
|
2021-11-19 16:10:07 +00:00
|
|
|
LUAU_FASTFLAG(LuauErrorRecoveryType);
|
2022-03-04 16:36:33 +00:00
|
|
|
LUAU_FASTFLAGVARIABLE(LuauSubtypingAddOptPropsToUnsealedTables, false)
|
2022-03-18 00:46:04 +00:00
|
|
|
LUAU_FASTFLAGVARIABLE(LuauTxnLogRefreshFunctionPointers, false)
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
namespace Luau
|
|
|
|
{
|
2021-12-10 22:05:05 +00:00
|
|
|
|
2022-05-06 01:03:43 +01:00
|
|
|
struct PromoteTypeLevels final : TypeVarOnceVisitor
|
2021-12-10 22:05:05 +00:00
|
|
|
{
|
|
|
|
TxnLog& log;
|
2022-02-11 19:02:09 +00:00
|
|
|
const TypeArena* typeArena = nullptr;
|
2021-12-10 22:05:05 +00:00
|
|
|
TypeLevel minLevel;
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
PromoteTypeLevels(TxnLog& log, const TypeArena* typeArena, TypeLevel minLevel)
|
|
|
|
: log(log)
|
2022-02-11 19:02:09 +00:00
|
|
|
, typeArena(typeArena)
|
2021-12-10 22:05:05 +00:00
|
|
|
, minLevel(minLevel)
|
2022-01-07 01:46:53 +00:00
|
|
|
{
|
|
|
|
}
|
2021-12-10 22:05:05 +00:00
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
template<typename TID, typename T>
|
2021-12-10 22:05:05 +00:00
|
|
|
void promote(TID ty, T* t)
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(t);
|
|
|
|
if (minLevel.subsumesStrict(t->level))
|
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
log.changeLevel(ty, minLevel);
|
2021-12-10 22:05:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-06 01:03:43 +01:00
|
|
|
// TODO cycle and operator() need to be clipped when FFlagLuauUseVisitRecursionLimit is clipped
|
2021-12-10 22:05:05 +00:00
|
|
|
template<typename TID>
|
2022-01-07 01:46:53 +00:00
|
|
|
void cycle(TID)
|
|
|
|
{
|
|
|
|
}
|
2021-12-10 22:05:05 +00:00
|
|
|
template<typename TID, typename T>
|
2022-02-11 19:02:09 +00:00
|
|
|
bool operator()(TID ty, const T&)
|
2022-05-06 01:03:43 +01:00
|
|
|
{
|
|
|
|
return visit(ty);
|
|
|
|
}
|
|
|
|
bool operator()(TypeId ty, const FreeTypeVar& ftv)
|
|
|
|
{
|
|
|
|
return visit(ty, ftv);
|
|
|
|
}
|
|
|
|
bool operator()(TypeId ty, const FunctionTypeVar& ftv)
|
|
|
|
{
|
|
|
|
return visit(ty, ftv);
|
|
|
|
}
|
|
|
|
bool operator()(TypeId ty, const TableTypeVar& ttv)
|
|
|
|
{
|
|
|
|
return visit(ty, ttv);
|
|
|
|
}
|
|
|
|
bool operator()(TypePackId tp, const FreeTypePack& ftp)
|
|
|
|
{
|
|
|
|
return visit(tp, ftp);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(TypeId ty) override
|
2021-12-10 22:05:05 +00:00
|
|
|
{
|
2022-02-11 19:02:09 +00:00
|
|
|
// Type levels of types from other modules are already global, so we don't need to promote anything inside
|
2022-03-24 22:04:14 +00:00
|
|
|
if (ty->owningArena != typeArena)
|
2022-02-11 19:02:09 +00:00
|
|
|
return false;
|
|
|
|
|
2021-12-10 22:05:05 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-05-06 01:03:43 +01:00
|
|
|
bool visit(TypePackId tp) override
|
|
|
|
{
|
|
|
|
// Type levels of types from other modules are already global, so we don't need to promote anything inside
|
|
|
|
if (tp->owningArena != typeArena)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(TypeId ty, const FreeTypeVar&) override
|
2021-12-10 22:05:05 +00:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
// Surprise, it's actually a BoundTypeVar that hasn't been committed yet.
|
|
|
|
// Calling getMutable on this will trigger an assertion.
|
2022-03-11 16:55:02 +00:00
|
|
|
if (!log.is<FreeTypeVar>(ty))
|
2022-01-07 01:46:53 +00:00
|
|
|
return true;
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
promote(ty, log.getMutable<FreeTypeVar>(ty));
|
2021-12-10 22:05:05 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-05-06 01:03:43 +01:00
|
|
|
bool visit(TypeId ty, const FunctionTypeVar&) override
|
2021-12-10 22:05:05 +00:00
|
|
|
{
|
2022-02-11 19:02:09 +00:00
|
|
|
// Type levels of types from other modules are already global, so we don't need to promote anything inside
|
2022-03-24 22:04:14 +00:00
|
|
|
if (ty->owningArena != typeArena)
|
2022-02-11 19:02:09 +00:00
|
|
|
return false;
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
promote(ty, log.getMutable<FunctionTypeVar>(ty));
|
2021-12-10 22:05:05 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-05-06 01:03:43 +01:00
|
|
|
bool visit(TypeId ty, const TableTypeVar& ttv) override
|
2021-12-10 22:05:05 +00:00
|
|
|
{
|
2022-02-11 19:02:09 +00:00
|
|
|
// Type levels of types from other modules are already global, so we don't need to promote anything inside
|
2022-03-24 22:04:14 +00:00
|
|
|
if (ty->owningArena != typeArena)
|
2022-02-11 19:02:09 +00:00
|
|
|
return false;
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
if (ttv.state != TableState::Free && ttv.state != TableState::Generic)
|
|
|
|
return true;
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
promote(ty, log.getMutable<TableTypeVar>(ty));
|
2021-12-10 22:05:05 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-05-06 01:03:43 +01:00
|
|
|
bool visit(TypePackId tp, const FreeTypePack&) override
|
2021-12-10 22:05:05 +00:00
|
|
|
{
|
2022-01-21 17:00:19 +00:00
|
|
|
// Surprise, it's actually a BoundTypePack that hasn't been committed yet.
|
|
|
|
// Calling getMutable on this will trigger an assertion.
|
2022-03-11 16:55:02 +00:00
|
|
|
if (!log.is<FreeTypePack>(tp))
|
2022-01-21 17:00:19 +00:00
|
|
|
return true;
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
promote(tp, log.getMutable<FreeTypePack>(tp));
|
2021-12-10 22:05:05 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
static void promoteTypeLevels(TxnLog& log, const TypeArena* typeArena, TypeLevel minLevel, TypeId ty)
|
2021-12-10 22:05:05 +00:00
|
|
|
{
|
2022-02-11 19:02:09 +00:00
|
|
|
// Type levels of types from other modules are already global, so we don't need to promote anything inside
|
2022-03-24 22:04:14 +00:00
|
|
|
if (ty->owningArena != typeArena)
|
2022-02-11 19:02:09 +00:00
|
|
|
return;
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
PromoteTypeLevels ptl{log, typeArena, minLevel};
|
2022-05-26 23:08:16 +01:00
|
|
|
ptl.traverse(ty);
|
2021-12-10 22:05:05 +00:00
|
|
|
}
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
void promoteTypeLevels(TxnLog& log, const TypeArena* typeArena, TypeLevel minLevel, TypePackId tp)
|
2021-12-10 22:05:05 +00:00
|
|
|
{
|
2022-02-11 19:02:09 +00:00
|
|
|
// Type levels of types from other modules are already global, so we don't need to promote anything inside
|
2022-03-24 22:04:14 +00:00
|
|
|
if (tp->owningArena != typeArena)
|
2022-02-11 19:02:09 +00:00
|
|
|
return;
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
PromoteTypeLevels ptl{log, typeArena, minLevel};
|
2022-05-26 23:08:16 +01:00
|
|
|
ptl.traverse(tp);
|
2021-12-10 22:05:05 +00:00
|
|
|
}
|
|
|
|
|
2022-05-06 01:03:43 +01:00
|
|
|
struct SkipCacheForType final : TypeVarOnceVisitor
|
2021-11-05 15:47:21 +00:00
|
|
|
{
|
2022-02-11 19:02:09 +00:00
|
|
|
SkipCacheForType(const DenseHashMap<TypeId, bool>& skipCacheForType, const TypeArena* typeArena)
|
2021-11-05 15:47:21 +00:00
|
|
|
: skipCacheForType(skipCacheForType)
|
2022-02-11 19:02:09 +00:00
|
|
|
, typeArena(typeArena)
|
2021-11-05 15:47:21 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2022-05-06 01:03:43 +01:00
|
|
|
bool visit(TypeId, const FreeTypeVar&) override
|
2021-11-05 15:47:21 +00:00
|
|
|
{
|
|
|
|
result = true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-05-06 01:03:43 +01:00
|
|
|
bool visit(TypeId, const BoundTypeVar&) override
|
2021-11-05 15:47:21 +00:00
|
|
|
{
|
|
|
|
result = true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-05-06 01:03:43 +01:00
|
|
|
bool visit(TypeId, const GenericTypeVar&) override
|
2021-11-05 15:47:21 +00:00
|
|
|
{
|
|
|
|
result = true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-05-06 01:03:43 +01:00
|
|
|
bool visit(TypeId ty, const TableTypeVar&) override
|
2021-11-05 15:47:21 +00:00
|
|
|
{
|
2022-02-11 19:02:09 +00:00
|
|
|
// Types from other modules don't contain mutable elements and are ok to cache
|
2022-03-24 22:04:14 +00:00
|
|
|
if (ty->owningArena != typeArena)
|
2022-02-11 19:02:09 +00:00
|
|
|
return false;
|
|
|
|
|
2021-11-05 15:47:21 +00:00
|
|
|
TableTypeVar& ttv = *getMutable<TableTypeVar>(ty);
|
|
|
|
|
|
|
|
if (ttv.boundTo)
|
|
|
|
{
|
|
|
|
result = true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ttv.state != TableState::Sealed)
|
|
|
|
{
|
|
|
|
result = true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-05-06 01:03:43 +01:00
|
|
|
bool visit(TypeId ty) override
|
2021-11-05 15:47:21 +00:00
|
|
|
{
|
2022-02-11 19:02:09 +00:00
|
|
|
// Types from other modules don't contain mutable elements and are ok to cache
|
2022-03-24 22:04:14 +00:00
|
|
|
if (ty->owningArena != typeArena)
|
2022-02-11 19:02:09 +00:00
|
|
|
return false;
|
|
|
|
|
2021-11-05 15:47:21 +00:00
|
|
|
const bool* prev = skipCacheForType.find(ty);
|
|
|
|
|
|
|
|
if (prev && *prev)
|
|
|
|
{
|
|
|
|
result = true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-05-06 01:03:43 +01:00
|
|
|
bool visit(TypePackId tp) override
|
2021-11-05 15:47:21 +00:00
|
|
|
{
|
2022-02-11 19:02:09 +00:00
|
|
|
// Types from other modules don't contain mutable elements and are ok to cache
|
2022-03-24 22:04:14 +00:00
|
|
|
if (tp->owningArena != typeArena)
|
2022-02-11 19:02:09 +00:00
|
|
|
return false;
|
|
|
|
|
2021-11-05 15:47:21 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-05-06 01:03:43 +01:00
|
|
|
bool visit(TypePackId tp, const FreeTypePack&) override
|
2021-11-05 15:47:21 +00:00
|
|
|
{
|
|
|
|
result = true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-05-06 01:03:43 +01:00
|
|
|
bool visit(TypePackId tp, const BoundTypePack&) override
|
2021-11-05 15:47:21 +00:00
|
|
|
{
|
|
|
|
result = true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-05-06 01:03:43 +01:00
|
|
|
bool visit(TypePackId tp, const GenericTypePack&) override
|
2021-11-05 15:47:21 +00:00
|
|
|
{
|
|
|
|
result = true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const DenseHashMap<TypeId, bool>& skipCacheForType;
|
2022-02-11 19:02:09 +00:00
|
|
|
const TypeArena* typeArena = nullptr;
|
2021-11-05 15:47:21 +00:00
|
|
|
bool result = false;
|
|
|
|
};
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-02-24 23:53:37 +00:00
|
|
|
bool Widen::isDirty(TypeId ty)
|
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
return log->is<SingletonTypeVar>(ty);
|
2022-02-24 23:53:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Widen::isDirty(TypePackId)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
TypeId Widen::clean(TypeId ty)
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(isDirty(ty));
|
2022-03-11 16:55:02 +00:00
|
|
|
auto stv = log->getMutable<SingletonTypeVar>(ty);
|
2022-02-24 23:53:37 +00:00
|
|
|
LUAU_ASSERT(stv);
|
|
|
|
|
|
|
|
if (get<StringSingleton>(stv))
|
|
|
|
return getSingletonTypes().stringType;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// If this assert trips, it's likely we now have number singletons.
|
|
|
|
LUAU_ASSERT(get<BooleanSingleton>(stv));
|
|
|
|
return getSingletonTypes().booleanType;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TypePackId Widen::clean(TypePackId)
|
|
|
|
{
|
|
|
|
throw std::runtime_error("Widen attempted to clean a dirty type pack?");
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Widen::ignoreChildren(TypeId ty)
|
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
return !log->is<UnionTypeVar>(ty);
|
2022-02-24 23:53:37 +00:00
|
|
|
}
|
|
|
|
|
2022-05-26 23:08:16 +01:00
|
|
|
TypeId Widen::operator()(TypeId ty)
|
|
|
|
{
|
|
|
|
return substitute(ty).value_or(ty);
|
|
|
|
}
|
|
|
|
|
|
|
|
TypePackId Widen::operator()(TypePackId tp)
|
|
|
|
{
|
|
|
|
return substitute(tp).value_or(tp);
|
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
static std::optional<TypeError> hasUnificationTooComplex(const ErrorVec& errors)
|
|
|
|
{
|
|
|
|
auto isUnificationTooComplex = [](const TypeError& te) {
|
|
|
|
return nullptr != get<UnificationTooComplex>(te);
|
|
|
|
};
|
|
|
|
|
|
|
|
auto it = std::find_if(errors.begin(), errors.end(), isUnificationTooComplex);
|
|
|
|
if (it == errors.end())
|
|
|
|
return std::nullopt;
|
|
|
|
else
|
|
|
|
return *it;
|
|
|
|
}
|
|
|
|
|
2021-12-10 22:05:05 +00:00
|
|
|
// Used for tagged union matching heuristic, returns first singleton type field
|
|
|
|
static std::optional<std::pair<Luau::Name, const SingletonTypeVar*>> getTableMatchTag(TypeId type)
|
|
|
|
{
|
2022-02-24 23:53:37 +00:00
|
|
|
if (auto ttv = getTableType(type))
|
2021-12-10 22:05:05 +00:00
|
|
|
{
|
2022-02-24 23:53:37 +00:00
|
|
|
for (auto&& [name, prop] : ttv->props)
|
2022-02-04 16:45:57 +00:00
|
|
|
{
|
2022-02-24 23:53:37 +00:00
|
|
|
if (auto sing = get<SingletonTypeVar>(follow(prop.type)))
|
|
|
|
return {{name, sing}};
|
2022-02-04 16:45:57 +00:00
|
|
|
}
|
2021-12-10 22:05:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
2022-04-15 00:57:43 +01:00
|
|
|
Unifier::Unifier(TypeArena* types, Mode mode, const Location& location, Variance variance, UnifierSharedState& sharedState, TxnLog* parentLog)
|
2021-10-29 21:25:12 +01:00
|
|
|
: types(types)
|
|
|
|
, mode(mode)
|
2022-01-07 01:46:53 +00:00
|
|
|
, log(parentLog)
|
2021-10-29 21:25:12 +01:00
|
|
|
, location(location)
|
|
|
|
, variance(variance)
|
2021-11-05 15:47:21 +00:00
|
|
|
, sharedState(sharedState)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2021-11-05 15:47:21 +00:00
|
|
|
LUAU_ASSERT(sharedState.iceHandler);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
void Unifier::tryUnify(TypeId subTy, TypeId superTy, bool isFunctionCall, bool isIntersection)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2021-12-03 06:41:04 +00:00
|
|
|
sharedState.counters.iterationCount = 0;
|
2021-11-05 02:34:35 +00:00
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
tryUnify_(subTy, superTy, isFunctionCall, isIntersection);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
void Unifier::tryUnify_(TypeId subTy, TypeId superTy, bool isFunctionCall, bool isIntersection)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-04-15 00:57:43 +01:00
|
|
|
RecursionLimiter _ra(&sharedState.counters.recursionCount,
|
|
|
|
FFlag::LuauAutocompleteDynamicLimits ? sharedState.counters.recursionLimit : FInt::LuauTypeInferRecursionLimit, "TypeId tryUnify_");
|
2021-11-05 02:34:35 +00:00
|
|
|
|
2021-12-03 06:41:04 +00:00
|
|
|
++sharedState.counters.iterationCount;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-04-15 00:57:43 +01:00
|
|
|
if (FFlag::LuauAutocompleteDynamicLimits)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-04-15 00:57:43 +01:00
|
|
|
if (sharedState.counters.iterationLimit > 0 && sharedState.counters.iterationLimit < sharedState.counters.iterationCount)
|
|
|
|
{
|
|
|
|
reportError(TypeError{location, UnificationTooComplex{}});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (FInt::LuauTypeInferIterationLimit > 0 && FInt::LuauTypeInferIterationLimit < sharedState.counters.iterationCount)
|
|
|
|
{
|
|
|
|
reportError(TypeError{location, UnificationTooComplex{}});
|
|
|
|
return;
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
superTy = log.follow(superTy);
|
|
|
|
subTy = log.follow(subTy);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
if (superTy == subTy)
|
|
|
|
return;
|
|
|
|
|
2022-04-15 00:57:43 +01:00
|
|
|
if (log.get<ConstrainedTypeVar>(superTy))
|
|
|
|
return tryUnifyWithConstrainedSuperTypeVar(subTy, superTy);
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
auto superFree = log.getMutable<FreeTypeVar>(superTy);
|
|
|
|
auto subFree = log.getMutable<FreeTypeVar>(subTy);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
if (superFree && subFree && superFree->level.subsumes(subFree->level))
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
occursCheck(subTy, superTy);
|
|
|
|
|
2021-11-19 16:10:07 +00:00
|
|
|
// The occurrence check might have caused superTy no longer to be a free type
|
2022-03-11 16:55:02 +00:00
|
|
|
bool occursFailed = bool(log.getMutable<ErrorTypeVar>(subTy));
|
2022-01-07 01:46:53 +00:00
|
|
|
|
|
|
|
if (!occursFailed)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
log.replace(subTy, BoundTypeVar(superTy));
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
2022-01-07 01:46:53 +00:00
|
|
|
else if (superFree && subFree)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
occursCheck(superTy, subTy);
|
2021-11-19 16:10:07 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
bool occursFailed = bool(log.getMutable<ErrorTypeVar>(superTy));
|
2022-01-07 01:46:53 +00:00
|
|
|
|
|
|
|
if (!occursFailed)
|
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
if (superFree->level.subsumes(subFree->level))
|
2022-01-07 01:46:53 +00:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
log.changeLevel(subTy, superFree->level);
|
2022-01-07 01:46:53 +00:00
|
|
|
}
|
2022-03-11 16:55:02 +00:00
|
|
|
|
|
|
|
log.replace(superTy, BoundTypeVar(subTy));
|
2021-11-19 16:10:07 +00:00
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
return;
|
|
|
|
}
|
2022-01-07 01:46:53 +00:00
|
|
|
else if (superFree)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-02-24 23:53:37 +00:00
|
|
|
TypeLevel superLevel = superFree->level;
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
occursCheck(superTy, subTy);
|
2022-03-11 16:55:02 +00:00
|
|
|
bool occursFailed = bool(log.getMutable<ErrorTypeVar>(superTy));
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
// Unification can't change the level of a generic.
|
2022-03-11 16:55:02 +00:00
|
|
|
auto subGeneric = log.getMutable<GenericTypeVar>(subTy);
|
2022-01-07 01:46:53 +00:00
|
|
|
if (subGeneric && !subGeneric->level.subsumes(superLevel))
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
// TODO: a more informative error message? CLI-39912
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, GenericError{"Generic subtype escaping scope"}});
|
2021-10-29 21:25:12 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-11-19 16:10:07 +00:00
|
|
|
// The occurrence check might have caused superTy no longer to be a free type
|
2022-01-07 01:46:53 +00:00
|
|
|
if (!occursFailed)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
promoteTypeLevels(log, types, superLevel, subTy);
|
2022-05-26 23:08:16 +01:00
|
|
|
|
|
|
|
Widen widen{types};
|
2022-03-11 16:55:02 +00:00
|
|
|
log.replace(superTy, BoundTypeVar(widen(subTy)));
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2021-11-19 16:10:07 +00:00
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
return;
|
|
|
|
}
|
2022-01-07 01:46:53 +00:00
|
|
|
else if (subFree)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
TypeLevel subLevel = subFree->level;
|
2021-12-10 22:05:05 +00:00
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
occursCheck(subTy, superTy);
|
2022-03-11 16:55:02 +00:00
|
|
|
bool occursFailed = bool(log.getMutable<ErrorTypeVar>(subTy));
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
// Unification can't change the level of a generic.
|
2022-03-11 16:55:02 +00:00
|
|
|
auto superGeneric = log.getMutable<GenericTypeVar>(superTy);
|
2022-01-07 01:46:53 +00:00
|
|
|
if (superGeneric && !superGeneric->level.subsumes(subFree->level))
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
// TODO: a more informative error message? CLI-39912
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, GenericError{"Generic supertype escaping scope"}});
|
2021-10-29 21:25:12 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
if (!occursFailed)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
promoteTypeLevels(log, types, subLevel, superTy);
|
|
|
|
log.replace(subTy, BoundTypeVar(superTy));
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (get<ErrorTypeVar>(superTy) || get<AnyTypeVar>(superTy))
|
2022-01-07 01:46:53 +00:00
|
|
|
return tryUnifyWithAny(subTy, superTy);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-04-15 00:57:43 +01:00
|
|
|
if (get<AnyTypeVar>(subTy))
|
|
|
|
{
|
|
|
|
if (anyIsTop)
|
|
|
|
{
|
|
|
|
reportError(TypeError{location, TypeMismatch{superTy, subTy}});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return tryUnifyWithAny(superTy, subTy);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (get<ErrorTypeVar>(subTy))
|
2022-01-07 01:46:53 +00:00
|
|
|
return tryUnifyWithAny(superTy, subTy);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2021-11-05 15:47:21 +00:00
|
|
|
auto& cache = sharedState.cachedUnify;
|
|
|
|
|
|
|
|
// What if the types are immutable and we proved their relation before
|
2022-04-21 22:44:27 +01:00
|
|
|
bool cacheEnabled = !isFunctionCall && !isIntersection && variance == Invariant;
|
2022-03-24 22:04:14 +00:00
|
|
|
|
2022-04-21 22:44:27 +01:00
|
|
|
if (cacheEnabled)
|
2022-03-24 22:04:14 +00:00
|
|
|
{
|
2022-04-21 22:44:27 +01:00
|
|
|
if (cache.contains({subTy, superTy}))
|
|
|
|
return;
|
2022-03-24 22:04:14 +00:00
|
|
|
|
2022-04-21 22:44:27 +01:00
|
|
|
if (auto error = sharedState.cachedUnifyError.find({subTy, superTy}))
|
|
|
|
{
|
|
|
|
reportError(TypeError{location, *error});
|
2022-03-24 22:04:14 +00:00
|
|
|
return;
|
2022-04-21 22:44:27 +01:00
|
|
|
}
|
2022-03-24 22:04:14 +00:00
|
|
|
}
|
2021-11-05 15:47:21 +00:00
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
// If we have seen this pair of types before, we are currently recursing into cyclic types.
|
|
|
|
// Here, we assume that the types unify. If they do not, we will find out as we roll back
|
|
|
|
// the stack.
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (log.haveSeen(superTy, subTy))
|
|
|
|
return;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
log.pushSeen(superTy, subTy);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-24 22:04:14 +00:00
|
|
|
size_t errorCount = errors.size();
|
|
|
|
|
2022-04-15 00:57:43 +01:00
|
|
|
if (log.get<ConstrainedTypeVar>(subTy))
|
|
|
|
tryUnifyWithConstrainedSubTypeVar(subTy, superTy);
|
|
|
|
else if (const UnionTypeVar* uv = log.getMutable<UnionTypeVar>(subTy))
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-02-04 16:45:57 +00:00
|
|
|
tryUnifyUnionWithType(subTy, uv, superTy);
|
|
|
|
}
|
2022-03-11 16:55:02 +00:00
|
|
|
else if (const UnionTypeVar* uv = log.getMutable<UnionTypeVar>(superTy))
|
2022-02-04 16:45:57 +00:00
|
|
|
{
|
|
|
|
tryUnifyTypeWithUnion(subTy, superTy, uv, cacheEnabled, isFunctionCall);
|
|
|
|
}
|
2022-03-11 16:55:02 +00:00
|
|
|
else if (const IntersectionTypeVar* uv = log.getMutable<IntersectionTypeVar>(superTy))
|
2022-02-04 16:45:57 +00:00
|
|
|
{
|
|
|
|
tryUnifyTypeWithIntersection(subTy, superTy, uv);
|
|
|
|
}
|
2022-03-11 16:55:02 +00:00
|
|
|
else if (const IntersectionTypeVar* uv = log.getMutable<IntersectionTypeVar>(subTy))
|
2022-02-04 16:45:57 +00:00
|
|
|
{
|
|
|
|
tryUnifyIntersectionWithType(subTy, uv, superTy, cacheEnabled, isFunctionCall);
|
|
|
|
}
|
2022-03-11 16:55:02 +00:00
|
|
|
else if (log.getMutable<PrimitiveTypeVar>(superTy) && log.getMutable<PrimitiveTypeVar>(subTy))
|
2022-02-04 16:45:57 +00:00
|
|
|
tryUnifyPrimitives(subTy, superTy);
|
|
|
|
|
2022-03-24 22:04:14 +00:00
|
|
|
else if ((log.getMutable<PrimitiveTypeVar>(superTy) || log.getMutable<SingletonTypeVar>(superTy)) && log.getMutable<SingletonTypeVar>(subTy))
|
2022-02-04 16:45:57 +00:00
|
|
|
tryUnifySingletons(subTy, superTy);
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
else if (log.getMutable<FunctionTypeVar>(superTy) && log.getMutable<FunctionTypeVar>(subTy))
|
2022-02-04 16:45:57 +00:00
|
|
|
tryUnifyFunctions(subTy, superTy, isFunctionCall);
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
else if (log.getMutable<TableTypeVar>(superTy) && log.getMutable<TableTypeVar>(subTy))
|
2022-02-04 16:45:57 +00:00
|
|
|
{
|
|
|
|
tryUnifyTables(subTy, superTy, isIntersection);
|
|
|
|
}
|
|
|
|
|
|
|
|
// tryUnifyWithMetatable assumes its first argument is a MetatableTypeVar. The check is otherwise symmetrical.
|
2022-03-11 16:55:02 +00:00
|
|
|
else if (log.getMutable<MetatableTypeVar>(superTy))
|
2022-02-04 16:45:57 +00:00
|
|
|
tryUnifyWithMetatable(subTy, superTy, /*reversed*/ false);
|
2022-03-11 16:55:02 +00:00
|
|
|
else if (log.getMutable<MetatableTypeVar>(subTy))
|
2022-02-04 16:45:57 +00:00
|
|
|
tryUnifyWithMetatable(superTy, subTy, /*reversed*/ true);
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
else if (log.getMutable<ClassTypeVar>(superTy))
|
2022-02-04 16:45:57 +00:00
|
|
|
tryUnifyWithClass(subTy, superTy, /*reversed*/ false);
|
|
|
|
|
|
|
|
// Unification of nonclasses with classes is almost, but not quite symmetrical.
|
|
|
|
// The order in which we perform this test is significant in the case that both types are classes.
|
2022-03-11 16:55:02 +00:00
|
|
|
else if (log.getMutable<ClassTypeVar>(subTy))
|
2022-02-04 16:45:57 +00:00
|
|
|
tryUnifyWithClass(subTy, superTy, /*reversed*/ true);
|
|
|
|
|
|
|
|
else
|
|
|
|
reportError(TypeError{location, TypeMismatch{superTy, subTy}});
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-04-21 22:44:27 +01:00
|
|
|
if (cacheEnabled)
|
2022-03-24 22:04:14 +00:00
|
|
|
cacheResult(subTy, superTy, errorCount);
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
log.popSeen(superTy, subTy);
|
2022-02-04 16:45:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Unifier::tryUnifyUnionWithType(TypeId subTy, const UnionTypeVar* uv, TypeId superTy)
|
|
|
|
{
|
|
|
|
// A | B <: T if A <: T and B <: T
|
|
|
|
bool failed = false;
|
|
|
|
std::optional<TypeError> unificationTooComplex;
|
|
|
|
std::optional<TypeError> firstFailedOption;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-02-04 16:45:57 +00:00
|
|
|
for (TypeId type : uv->options)
|
|
|
|
{
|
|
|
|
Unifier innerState = makeChildUnifier();
|
|
|
|
innerState.tryUnify_(type, superTy);
|
|
|
|
|
|
|
|
if (auto e = hasUnificationTooComplex(innerState.errors))
|
|
|
|
unificationTooComplex = e;
|
|
|
|
else if (!innerState.errors.empty())
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-02-04 16:45:57 +00:00
|
|
|
// 'nil' option is skipped from extended report because we present the type in a special way - 'T?'
|
|
|
|
if (!firstFailedOption && !isNil(type))
|
|
|
|
firstFailedOption = {innerState.errors.front()};
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-02-04 16:45:57 +00:00
|
|
|
failed = true;
|
|
|
|
}
|
2022-03-04 16:36:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// even if A | B <: T fails, we want to bind some options of T with A | B iff A | B was a subtype of that option.
|
2022-05-26 23:08:16 +01:00
|
|
|
auto tryBind = [this, subTy](TypeId superOption) {
|
|
|
|
superOption = log.follow(superOption);
|
2022-03-04 16:36:33 +00:00
|
|
|
|
2022-05-26 23:08:16 +01:00
|
|
|
// just skip if the superOption is not free-ish.
|
|
|
|
auto ttv = log.getMutable<TableTypeVar>(superOption);
|
|
|
|
if (!log.is<FreeTypeVar>(superOption) && (!ttv || ttv->state != TableState::Free))
|
|
|
|
return;
|
2022-05-06 01:03:43 +01:00
|
|
|
|
2022-05-26 23:08:16 +01:00
|
|
|
// If superOption is already present in subTy, do nothing. Nothing new has been learned, but the subtype
|
|
|
|
// test is successful.
|
|
|
|
if (auto subUnion = get<UnionTypeVar>(subTy))
|
|
|
|
{
|
|
|
|
if (end(subUnion) != std::find(begin(subUnion), end(subUnion), superOption))
|
|
|
|
return;
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-05-26 23:08:16 +01:00
|
|
|
// Since we have already checked if S <: T, checking it again will not queue up the type for replacement.
|
|
|
|
// So we'll have to do it ourselves. We assume they unified cleanly if they are still in the seen set.
|
|
|
|
if (log.haveSeen(subTy, superOption))
|
2022-03-04 16:36:33 +00:00
|
|
|
{
|
2022-05-26 23:08:16 +01:00
|
|
|
// TODO: would it be nice for TxnLog::replace to do this?
|
|
|
|
if (log.is<TableTypeVar>(superOption))
|
|
|
|
log.bindTable(superOption, subTy);
|
|
|
|
else
|
|
|
|
log.replace(superOption, *subTy);
|
2022-03-04 16:36:33 +00:00
|
|
|
}
|
2022-05-26 23:08:16 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
if (auto utv = log.getMutable<UnionTypeVar>(superTy))
|
|
|
|
{
|
|
|
|
for (TypeId ty : utv)
|
|
|
|
tryBind(ty);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-05-26 23:08:16 +01:00
|
|
|
else
|
|
|
|
tryBind(superTy);
|
2022-02-04 16:45:57 +00:00
|
|
|
|
|
|
|
if (unificationTooComplex)
|
|
|
|
reportError(*unificationTooComplex);
|
|
|
|
else if (failed)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-02-04 16:45:57 +00:00
|
|
|
if (firstFailedOption)
|
|
|
|
reportError(TypeError{location, TypeMismatch{superTy, subTy, "Not all union options are compatible.", *firstFailedOption}});
|
|
|
|
else
|
|
|
|
reportError(TypeError{location, TypeMismatch{superTy, subTy}});
|
|
|
|
}
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-02-04 16:45:57 +00:00
|
|
|
void Unifier::tryUnifyTypeWithUnion(TypeId subTy, TypeId superTy, const UnionTypeVar* uv, bool cacheEnabled, bool isFunctionCall)
|
|
|
|
{
|
|
|
|
// T <: A | B if T <: A or T <: B
|
|
|
|
bool found = false;
|
|
|
|
std::optional<TypeError> unificationTooComplex;
|
2021-12-10 22:05:05 +00:00
|
|
|
|
2022-02-04 16:45:57 +00:00
|
|
|
size_t failedOptionCount = 0;
|
|
|
|
std::optional<TypeError> failedOption;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-02-04 16:45:57 +00:00
|
|
|
bool foundHeuristic = false;
|
|
|
|
size_t startIndex = 0;
|
2021-11-05 15:47:21 +00:00
|
|
|
|
2022-02-04 16:45:57 +00:00
|
|
|
if (const std::string* subName = getName(subTy))
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < uv->options.size(); ++i)
|
2022-01-27 23:46:05 +00:00
|
|
|
{
|
2022-02-04 16:45:57 +00:00
|
|
|
const std::string* optionName = getName(uv->options[i]);
|
|
|
|
if (optionName && *optionName == *subName)
|
2021-12-10 22:05:05 +00:00
|
|
|
{
|
2022-02-04 16:45:57 +00:00
|
|
|
foundHeuristic = true;
|
|
|
|
startIndex = i;
|
|
|
|
break;
|
2021-12-10 22:05:05 +00:00
|
|
|
}
|
2022-01-27 23:46:05 +00:00
|
|
|
}
|
2022-02-04 16:45:57 +00:00
|
|
|
}
|
2021-12-10 22:05:05 +00:00
|
|
|
|
2022-02-04 16:45:57 +00:00
|
|
|
if (auto subMatchTag = getTableMatchTag(subTy))
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < uv->options.size(); ++i)
|
2022-01-27 23:46:05 +00:00
|
|
|
{
|
2022-02-04 16:45:57 +00:00
|
|
|
auto optionMatchTag = getTableMatchTag(uv->options[i]);
|
|
|
|
if (optionMatchTag && optionMatchTag->first == subMatchTag->first && *optionMatchTag->second == *subMatchTag->second)
|
2021-11-05 15:47:21 +00:00
|
|
|
{
|
2022-02-04 16:45:57 +00:00
|
|
|
foundHeuristic = true;
|
|
|
|
startIndex = i;
|
|
|
|
break;
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
2022-02-04 16:45:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!foundHeuristic && cacheEnabled)
|
|
|
|
{
|
|
|
|
auto& cache = sharedState.cachedUnify;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
for (size_t i = 0; i < uv->options.size(); ++i)
|
|
|
|
{
|
2022-02-04 16:45:57 +00:00
|
|
|
TypeId type = uv->options[i];
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-04-21 22:44:27 +01:00
|
|
|
if (cache.contains({subTy, type}))
|
2022-03-24 22:04:14 +00:00
|
|
|
{
|
2022-04-21 22:44:27 +01:00
|
|
|
startIndex = i;
|
|
|
|
break;
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-02-04 16:45:57 +00:00
|
|
|
}
|
|
|
|
}
|
2021-12-10 22:05:05 +00:00
|
|
|
|
2022-02-04 16:45:57 +00:00
|
|
|
for (size_t i = 0; i < uv->options.size(); ++i)
|
|
|
|
{
|
|
|
|
TypeId type = uv->options[(i + startIndex) % uv->options.size()];
|
|
|
|
Unifier innerState = makeChildUnifier();
|
|
|
|
innerState.tryUnify_(subTy, type, isFunctionCall);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-02-04 16:45:57 +00:00
|
|
|
if (innerState.errors.empty())
|
|
|
|
{
|
|
|
|
found = true;
|
2022-03-11 16:55:02 +00:00
|
|
|
log.concat(std::move(innerState.log));
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-02-04 16:45:57 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
else if (auto e = hasUnificationTooComplex(innerState.errors))
|
2021-12-10 22:05:05 +00:00
|
|
|
{
|
2022-02-04 16:45:57 +00:00
|
|
|
unificationTooComplex = e;
|
2021-12-10 22:05:05 +00:00
|
|
|
}
|
2022-02-04 16:45:57 +00:00
|
|
|
else if (!isNil(type))
|
2021-11-12 14:27:34 +00:00
|
|
|
{
|
2022-02-04 16:45:57 +00:00
|
|
|
failedOptionCount++;
|
|
|
|
|
|
|
|
if (!failedOption)
|
|
|
|
failedOption = {innerState.errors.front()};
|
2021-11-12 14:27:34 +00:00
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-02-04 16:45:57 +00:00
|
|
|
|
|
|
|
if (unificationTooComplex)
|
|
|
|
{
|
|
|
|
reportError(*unificationTooComplex);
|
|
|
|
}
|
|
|
|
else if (!found)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-02-04 16:45:57 +00:00
|
|
|
if ((failedOptionCount == 1 || foundHeuristic) && failedOption)
|
|
|
|
reportError(TypeError{location, TypeMismatch{superTy, subTy, "None of the union options are compatible. For example:", *failedOption}});
|
|
|
|
else
|
|
|
|
reportError(TypeError{location, TypeMismatch{superTy, subTy, "none of the union options are compatible"}});
|
|
|
|
}
|
|
|
|
}
|
2022-01-06 23:26:14 +00:00
|
|
|
|
2022-02-04 16:45:57 +00:00
|
|
|
void Unifier::tryUnifyTypeWithIntersection(TypeId subTy, TypeId superTy, const IntersectionTypeVar* uv)
|
|
|
|
{
|
|
|
|
std::optional<TypeError> unificationTooComplex;
|
|
|
|
std::optional<TypeError> firstFailedOption;
|
2021-11-12 14:27:34 +00:00
|
|
|
|
2022-02-04 16:45:57 +00:00
|
|
|
// T <: A & B if A <: T and B <: T
|
|
|
|
for (TypeId type : uv->parts)
|
|
|
|
{
|
|
|
|
Unifier innerState = makeChildUnifier();
|
|
|
|
innerState.tryUnify_(subTy, type, /*isFunctionCall*/ false, /*isIntersection*/ true);
|
2021-11-12 14:27:34 +00:00
|
|
|
|
2022-02-04 16:45:57 +00:00
|
|
|
if (auto e = hasUnificationTooComplex(innerState.errors))
|
|
|
|
unificationTooComplex = e;
|
|
|
|
else if (!innerState.errors.empty())
|
|
|
|
{
|
|
|
|
if (!firstFailedOption)
|
|
|
|
firstFailedOption = {innerState.errors.front()};
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-01-06 23:26:14 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
log.concat(std::move(innerState.log));
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-02-04 16:45:57 +00:00
|
|
|
if (unificationTooComplex)
|
|
|
|
reportError(*unificationTooComplex);
|
|
|
|
else if (firstFailedOption)
|
|
|
|
reportError(TypeError{location, TypeMismatch{superTy, subTy, "Not all intersection parts are compatible.", *firstFailedOption}});
|
|
|
|
}
|
2021-11-05 15:47:21 +00:00
|
|
|
|
2022-02-04 16:45:57 +00:00
|
|
|
void Unifier::tryUnifyIntersectionWithType(TypeId subTy, const IntersectionTypeVar* uv, TypeId superTy, bool cacheEnabled, bool isFunctionCall)
|
|
|
|
{
|
|
|
|
// A & B <: T if T <: A or T <: B
|
|
|
|
bool found = false;
|
|
|
|
std::optional<TypeError> unificationTooComplex;
|
2021-11-05 15:47:21 +00:00
|
|
|
|
2022-02-04 16:45:57 +00:00
|
|
|
size_t startIndex = 0;
|
|
|
|
|
|
|
|
if (cacheEnabled)
|
|
|
|
{
|
|
|
|
auto& cache = sharedState.cachedUnify;
|
2021-11-05 15:47:21 +00:00
|
|
|
|
|
|
|
for (size_t i = 0; i < uv->parts.size(); ++i)
|
|
|
|
{
|
2022-02-04 16:45:57 +00:00
|
|
|
TypeId type = uv->parts[i];
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-04-21 22:44:27 +01:00
|
|
|
if (cache.contains({type, superTy}))
|
2022-03-24 22:04:14 +00:00
|
|
|
{
|
2022-04-21 22:44:27 +01:00
|
|
|
startIndex = i;
|
|
|
|
break;
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
2022-02-04 16:45:57 +00:00
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-02-04 16:45:57 +00:00
|
|
|
for (size_t i = 0; i < uv->parts.size(); ++i)
|
|
|
|
{
|
|
|
|
TypeId type = uv->parts[(i + startIndex) % uv->parts.size()];
|
|
|
|
Unifier innerState = makeChildUnifier();
|
|
|
|
innerState.tryUnify_(type, superTy, isFunctionCall);
|
|
|
|
|
|
|
|
if (innerState.errors.empty())
|
2021-11-12 14:27:34 +00:00
|
|
|
{
|
2022-02-04 16:45:57 +00:00
|
|
|
found = true;
|
2022-03-11 16:55:02 +00:00
|
|
|
log.concat(std::move(innerState.log));
|
2022-02-04 16:45:57 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
else if (auto e = hasUnificationTooComplex(innerState.errors))
|
|
|
|
{
|
|
|
|
unificationTooComplex = e;
|
2021-11-12 14:27:34 +00:00
|
|
|
}
|
2022-02-04 16:45:57 +00:00
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-02-04 16:45:57 +00:00
|
|
|
if (unificationTooComplex)
|
|
|
|
reportError(*unificationTooComplex);
|
|
|
|
else if (!found)
|
2021-11-05 15:47:21 +00:00
|
|
|
{
|
2022-02-04 16:45:57 +00:00
|
|
|
reportError(TypeError{location, TypeMismatch{superTy, subTy, "none of the intersection parts are compatible"}});
|
2021-11-05 15:47:21 +00:00
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-03-24 22:04:14 +00:00
|
|
|
bool Unifier::canCacheResult(TypeId subTy, TypeId superTy)
|
2021-11-05 15:47:21 +00:00
|
|
|
{
|
|
|
|
bool* superTyInfo = sharedState.skipCacheForType.find(superTy);
|
|
|
|
|
|
|
|
if (superTyInfo && *superTyInfo)
|
2022-03-24 22:04:14 +00:00
|
|
|
return false;
|
2021-11-05 15:47:21 +00:00
|
|
|
|
|
|
|
bool* subTyInfo = sharedState.skipCacheForType.find(subTy);
|
|
|
|
|
|
|
|
if (subTyInfo && *subTyInfo)
|
2022-03-24 22:04:14 +00:00
|
|
|
return false;
|
2021-11-05 15:47:21 +00:00
|
|
|
|
|
|
|
auto skipCacheFor = [this](TypeId ty) {
|
2022-02-11 19:02:09 +00:00
|
|
|
SkipCacheForType visitor{sharedState.skipCacheForType, types};
|
2022-05-26 23:08:16 +01:00
|
|
|
visitor.traverse(ty);
|
2021-11-05 15:47:21 +00:00
|
|
|
|
|
|
|
sharedState.skipCacheForType[ty] = visitor.result;
|
|
|
|
|
|
|
|
return visitor.result;
|
|
|
|
};
|
|
|
|
|
|
|
|
if (!superTyInfo && skipCacheFor(superTy))
|
2022-03-24 22:04:14 +00:00
|
|
|
return false;
|
2021-11-05 15:47:21 +00:00
|
|
|
|
|
|
|
if (!subTyInfo && skipCacheFor(subTy))
|
2022-03-24 22:04:14 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Unifier::cacheResult(TypeId subTy, TypeId superTy, size_t prevErrorCount)
|
|
|
|
{
|
|
|
|
if (errors.size() == prevErrorCount)
|
|
|
|
{
|
|
|
|
if (canCacheResult(subTy, superTy))
|
|
|
|
sharedState.cachedUnify.insert({subTy, superTy});
|
|
|
|
}
|
|
|
|
else if (errors.size() == prevErrorCount + 1)
|
|
|
|
{
|
|
|
|
if (canCacheResult(subTy, superTy))
|
|
|
|
sharedState.cachedUnifyError[{subTy, superTy}] = errors.back().data;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
struct WeirdIter
|
|
|
|
{
|
|
|
|
TypePackId packId;
|
|
|
|
TxnLog& log;
|
|
|
|
TypePack* pack;
|
|
|
|
size_t index;
|
|
|
|
bool growing;
|
|
|
|
TypeLevel level;
|
|
|
|
|
|
|
|
WeirdIter(TypePackId packId, TxnLog& log)
|
|
|
|
: packId(packId)
|
|
|
|
, log(log)
|
|
|
|
, pack(log.getMutable<TypePack>(packId))
|
|
|
|
, index(0)
|
|
|
|
, growing(false)
|
|
|
|
{
|
|
|
|
while (pack && pack->head.empty() && pack->tail)
|
|
|
|
{
|
|
|
|
packId = *pack->tail;
|
|
|
|
pack = log.getMutable<TypePack>(packId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
WeirdIter(const WeirdIter&) = default;
|
|
|
|
|
|
|
|
TypeId& operator*()
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(good());
|
|
|
|
return pack->head[index];
|
|
|
|
}
|
|
|
|
|
|
|
|
bool good() const
|
|
|
|
{
|
|
|
|
return pack != nullptr && index < pack->head.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool advance()
|
|
|
|
{
|
|
|
|
if (!pack)
|
|
|
|
return good();
|
|
|
|
|
|
|
|
if (index < pack->head.size())
|
|
|
|
++index;
|
|
|
|
|
|
|
|
if (growing || index < pack->head.size())
|
|
|
|
return good();
|
|
|
|
|
|
|
|
if (pack->tail)
|
|
|
|
{
|
|
|
|
packId = log.follow(*pack->tail);
|
|
|
|
pack = log.getMutable<TypePack>(packId);
|
|
|
|
index = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return good();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool canGrow() const
|
|
|
|
{
|
|
|
|
return nullptr != log.getMutable<Unifiable::Free>(packId);
|
|
|
|
}
|
|
|
|
|
|
|
|
void grow(TypePackId newTail)
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(canGrow());
|
|
|
|
LUAU_ASSERT(log.getMutable<TypePack>(newTail));
|
|
|
|
|
|
|
|
level = log.getMutable<Unifiable::Free>(packId)->level;
|
2022-04-15 00:57:43 +01:00
|
|
|
log.replace(packId, BoundTypePack(newTail));
|
2022-01-07 01:46:53 +00:00
|
|
|
packId = newTail;
|
|
|
|
pack = log.getMutable<TypePack>(newTail);
|
|
|
|
index = 0;
|
|
|
|
growing = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void pushType(TypeId ty)
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(pack);
|
|
|
|
PendingTypePack* pendingPack = log.queue(packId);
|
|
|
|
if (TypePack* pending = getMutable<TypePack>(pendingPack))
|
|
|
|
{
|
|
|
|
pending->head.push_back(ty);
|
|
|
|
// We've potentially just replaced the TypePack* that we need to look
|
|
|
|
// in. We need to replace pack.
|
|
|
|
pack = pending;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(!"Pending state for this pack was not a TypePack");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
ErrorVec Unifier::canUnify(TypeId subTy, TypeId superTy)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
Unifier s = makeChildUnifier();
|
2022-01-07 01:46:53 +00:00
|
|
|
s.tryUnify_(subTy, superTy);
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
return s.errors;
|
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
ErrorVec Unifier::canUnify(TypePackId subTy, TypePackId superTy, bool isFunctionCall)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
Unifier s = makeChildUnifier();
|
2022-01-07 01:46:53 +00:00
|
|
|
s.tryUnify_(subTy, superTy, isFunctionCall);
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
return s.errors;
|
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
void Unifier::tryUnify(TypePackId subTp, TypePackId superTp, bool isFunctionCall)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2021-12-03 06:41:04 +00:00
|
|
|
sharedState.counters.iterationCount = 0;
|
2021-11-05 02:34:35 +00:00
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
tryUnify_(subTp, superTp, isFunctionCall);
|
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
/*
|
|
|
|
* This is quite tricky: we are walking two rope-like structures and unifying corresponding elements.
|
|
|
|
* If one is longer than the other, but the short end is free, we grow it to the required length.
|
|
|
|
*/
|
2022-01-07 01:46:53 +00:00
|
|
|
void Unifier::tryUnify_(TypePackId subTp, TypePackId superTp, bool isFunctionCall)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-04-15 00:57:43 +01:00
|
|
|
RecursionLimiter _ra(&sharedState.counters.recursionCount,
|
|
|
|
FFlag::LuauAutocompleteDynamicLimits ? sharedState.counters.recursionLimit : FInt::LuauTypeInferRecursionLimit, "TypePackId tryUnify_");
|
2021-11-05 02:34:35 +00:00
|
|
|
|
2021-12-03 06:41:04 +00:00
|
|
|
++sharedState.counters.iterationCount;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-04-15 00:57:43 +01:00
|
|
|
if (FFlag::LuauAutocompleteDynamicLimits)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-04-15 00:57:43 +01:00
|
|
|
if (sharedState.counters.iterationLimit > 0 && sharedState.counters.iterationLimit < sharedState.counters.iterationCount)
|
|
|
|
{
|
|
|
|
reportError(TypeError{location, UnificationTooComplex{}});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (FInt::LuauTypeInferIterationLimit > 0 && FInt::LuauTypeInferIterationLimit < sharedState.counters.iterationCount)
|
|
|
|
{
|
|
|
|
reportError(TypeError{location, UnificationTooComplex{}});
|
|
|
|
return;
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
superTp = log.follow(superTp);
|
|
|
|
subTp = log.follow(subTp);
|
|
|
|
|
|
|
|
while (auto tp = log.getMutable<TypePack>(subTp))
|
|
|
|
{
|
|
|
|
if (tp->head.empty() && tp->tail)
|
|
|
|
subTp = log.follow(*tp->tail);
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (auto tp = log.getMutable<TypePack>(superTp))
|
|
|
|
{
|
|
|
|
if (tp->head.empty() && tp->tail)
|
|
|
|
superTp = log.follow(*tp->tail);
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (superTp == subTp)
|
|
|
|
return;
|
|
|
|
|
2022-04-29 02:24:24 +01:00
|
|
|
if (log.haveSeen(superTp, subTp))
|
2022-03-11 16:55:02 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
if (log.getMutable<Unifiable::Free>(superTp))
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
occursCheck(superTp, subTp);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (!log.getMutable<ErrorTypeVar>(superTp))
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-05-26 23:08:16 +01:00
|
|
|
Widen widen{types};
|
2022-03-18 00:46:04 +00:00
|
|
|
log.replace(superTp, Unifiable::Bound<TypePackId>(widen(subTp)));
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-03-11 16:55:02 +00:00
|
|
|
}
|
|
|
|
else if (log.getMutable<Unifiable::Free>(subTp))
|
|
|
|
{
|
|
|
|
occursCheck(subTp, superTp);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (!log.getMutable<ErrorTypeVar>(subTp))
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
log.replace(subTp, Unifiable::Bound<TypePackId>(superTp));
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-03-11 16:55:02 +00:00
|
|
|
}
|
|
|
|
else if (log.getMutable<Unifiable::Error>(superTp))
|
|
|
|
tryUnifyWithAny(subTp, superTp);
|
|
|
|
else if (log.getMutable<Unifiable::Error>(subTp))
|
|
|
|
tryUnifyWithAny(superTp, subTp);
|
|
|
|
else if (log.getMutable<VariadicTypePack>(superTp))
|
|
|
|
tryUnifyVariadics(subTp, superTp, false);
|
|
|
|
else if (log.getMutable<VariadicTypePack>(subTp))
|
|
|
|
tryUnifyVariadics(superTp, subTp, true);
|
|
|
|
else if (log.getMutable<TypePack>(superTp) && log.getMutable<TypePack>(subTp))
|
|
|
|
{
|
|
|
|
auto superTpv = log.getMutable<TypePack>(superTp);
|
|
|
|
auto subTpv = log.getMutable<TypePack>(subTp);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
// If the size of two heads does not match, but both packs have free tail
|
|
|
|
// We set the sentinel variable to say so to avoid growing it forever.
|
2022-04-15 00:57:43 +01:00
|
|
|
auto [superTypes, superTail] = flatten(superTp, log);
|
|
|
|
auto [subTypes, subTail] = flatten(subTp, log);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
bool noInfiniteGrowth = (superTypes.size() != subTypes.size()) && (superTail && log.getMutable<FreeTypePack>(*superTail)) &&
|
|
|
|
(subTail && log.getMutable<FreeTypePack>(*subTail));
|
|
|
|
|
|
|
|
auto superIter = WeirdIter(superTp, log);
|
|
|
|
auto subIter = WeirdIter(subTp, log);
|
2022-03-04 16:36:33 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
auto mkFreshType = [this](TypeLevel level) {
|
|
|
|
return types->freshType(level);
|
|
|
|
};
|
|
|
|
|
|
|
|
const TypePackId emptyTp = types->addTypePack(TypePack{{}, std::nullopt});
|
|
|
|
|
|
|
|
int loopCount = 0;
|
|
|
|
|
|
|
|
do
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
if (FInt::LuauTypeInferTypePackLoopLimit > 0 && loopCount >= FInt::LuauTypeInferTypePackLoopLimit)
|
|
|
|
ice("Detected possibly infinite TypePack growth");
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
++loopCount;
|
|
|
|
|
|
|
|
if (superIter.good() && subIter.growing)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
subIter.pushType(mkFreshType(subIter.level));
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (subIter.good() && superIter.growing)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
superIter.pushType(mkFreshType(superIter.level));
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (superIter.good() && subIter.good())
|
|
|
|
{
|
|
|
|
tryUnify_(*subIter, *superIter);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (!errors.empty() && !firstPackErrorPos)
|
|
|
|
firstPackErrorPos = loopCount;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
superIter.advance();
|
|
|
|
subIter.advance();
|
|
|
|
continue;
|
|
|
|
}
|
2022-01-07 01:46:53 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
// If both are at the end, we're done
|
|
|
|
if (!superIter.good() && !subIter.good())
|
|
|
|
{
|
|
|
|
if (subTpv->tail && superTpv->tail)
|
|
|
|
{
|
|
|
|
tryUnify_(*subTpv->tail, *superTpv->tail);
|
|
|
|
break;
|
|
|
|
}
|
2022-01-07 01:46:53 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
const bool lFreeTail = superTpv->tail && log.getMutable<FreeTypePack>(log.follow(*superTpv->tail)) != nullptr;
|
|
|
|
const bool rFreeTail = subTpv->tail && log.getMutable<FreeTypePack>(log.follow(*subTpv->tail)) != nullptr;
|
|
|
|
if (lFreeTail)
|
|
|
|
tryUnify_(emptyTp, *superTpv->tail);
|
|
|
|
else if (rFreeTail)
|
|
|
|
tryUnify_(emptyTp, *subTpv->tail);
|
2022-01-07 01:46:53 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
break;
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
// If both tails are free, bind one to the other and call it a day
|
|
|
|
if (superIter.canGrow() && subIter.canGrow())
|
|
|
|
return tryUnify_(*subIter.pack->tail, *superIter.pack->tail);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
// If just one side is free on its tail, grow it to fit the other side.
|
|
|
|
// FIXME: The tail-most tail of the growing pack should be the same as the tail-most tail of the non-growing pack.
|
|
|
|
if (superIter.canGrow())
|
|
|
|
superIter.grow(types->addTypePack(TypePackVar(TypePack{})));
|
|
|
|
else if (subIter.canGrow())
|
|
|
|
subIter.grow(types->addTypePack(TypePackVar(TypePack{})));
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// A union type including nil marks an optional argument
|
2022-04-15 00:57:43 +01:00
|
|
|
if ((!FFlag::LuauLowerBoundsCalculation || isNonstrictMode()) && superIter.good() && isOptional(*superIter))
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
superIter.advance();
|
|
|
|
continue;
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-04-15 00:57:43 +01:00
|
|
|
else if ((!FFlag::LuauLowerBoundsCalculation || isNonstrictMode()) && subIter.good() && isOptional(*subIter))
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
subIter.advance();
|
|
|
|
continue;
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (log.getMutable<VariadicTypePack>(superIter.packId))
|
|
|
|
{
|
|
|
|
tryUnifyVariadics(subIter.packId, superIter.packId, false, int(subIter.index));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (log.getMutable<VariadicTypePack>(subIter.packId))
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
tryUnifyVariadics(superIter.packId, subIter.packId, true, int(superIter.index));
|
|
|
|
return;
|
|
|
|
}
|
2022-01-07 01:46:53 +00:00
|
|
|
|
2022-04-15 00:57:43 +01:00
|
|
|
if ((!FFlag::LuauLowerBoundsCalculation || isNonstrictMode()) && !isFunctionCall && subIter.good())
|
2022-03-11 16:55:02 +00:00
|
|
|
{
|
|
|
|
// Sometimes it is ok to pass too many arguments
|
|
|
|
return;
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
// This is a bit weird because we don't actually know expected vs actual. We just know
|
|
|
|
// subtype vs supertype. If we are checking the values returned by a function, we swap
|
|
|
|
// these to produce the expected error message.
|
|
|
|
size_t expectedSize = size(superTp);
|
|
|
|
size_t actualSize = size(subTp);
|
|
|
|
if (ctx == CountMismatch::Result)
|
|
|
|
std::swap(expectedSize, actualSize);
|
|
|
|
reportError(TypeError{location, CountMismatch{expectedSize, actualSize, ctx}});
|
2022-01-07 01:46:53 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
while (superIter.good())
|
2022-01-07 01:46:53 +00:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
tryUnify_(*superIter, getSingletonTypes().errorRecoveryType());
|
|
|
|
superIter.advance();
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
while (subIter.good())
|
|
|
|
{
|
|
|
|
tryUnify_(*subIter, getSingletonTypes().errorRecoveryType());
|
|
|
|
subIter.advance();
|
2022-01-07 01:46:53 +00:00
|
|
|
}
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
} while (!noInfiniteGrowth);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
reportError(TypeError{location, GenericError{"Failed to unify type packs"}});
|
|
|
|
}
|
|
|
|
}
|
2022-01-07 01:46:53 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
void Unifier::tryUnifyPrimitives(TypeId subTy, TypeId superTy)
|
|
|
|
{
|
|
|
|
const PrimitiveTypeVar* superPrim = get<PrimitiveTypeVar>(superTy);
|
|
|
|
const PrimitiveTypeVar* subPrim = get<PrimitiveTypeVar>(subTy);
|
|
|
|
if (!superPrim || !subPrim)
|
|
|
|
ice("passed non primitive types to unifyPrimitives");
|
2022-01-07 01:46:53 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (superPrim->type != subPrim->type)
|
|
|
|
reportError(TypeError{location, TypeMismatch{superTy, subTy}});
|
|
|
|
}
|
2022-01-07 01:46:53 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
void Unifier::tryUnifySingletons(TypeId subTy, TypeId superTy)
|
|
|
|
{
|
|
|
|
const PrimitiveTypeVar* superPrim = get<PrimitiveTypeVar>(superTy);
|
|
|
|
const SingletonTypeVar* superSingleton = get<SingletonTypeVar>(superTy);
|
|
|
|
const SingletonTypeVar* subSingleton = get<SingletonTypeVar>(subTy);
|
2022-01-07 01:46:53 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if ((!superPrim && !superSingleton) || !subSingleton)
|
|
|
|
ice("passed non singleton/primitive types to unifySingletons");
|
2021-11-19 16:10:07 +00:00
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
if (superSingleton && *superSingleton == *subSingleton)
|
2021-11-19 16:10:07 +00:00
|
|
|
return;
|
|
|
|
|
2022-01-27 23:46:05 +00:00
|
|
|
if (superPrim && superPrim->type == PrimitiveTypeVar::Boolean && get<BooleanSingleton>(subSingleton) && variance == Covariant)
|
2021-11-19 16:10:07 +00:00
|
|
|
return;
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
if (superPrim && superPrim->type == PrimitiveTypeVar::String && get<StringSingleton>(subSingleton) && variance == Covariant)
|
2021-11-19 16:10:07 +00:00
|
|
|
return;
|
|
|
|
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, TypeMismatch{superTy, subTy}});
|
2021-11-19 16:10:07 +00:00
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
void Unifier::tryUnifyFunctions(TypeId subTy, TypeId superTy, bool isFunctionCall)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
FunctionTypeVar* superFunction = log.getMutable<FunctionTypeVar>(superTy);
|
|
|
|
FunctionTypeVar* subFunction = log.getMutable<FunctionTypeVar>(subTy);
|
2022-01-07 01:46:53 +00:00
|
|
|
|
|
|
|
if (!superFunction || !subFunction)
|
2021-10-29 21:25:12 +01:00
|
|
|
ice("passed non-function types to unifyFunction");
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
size_t numGenerics = superFunction->generics.size();
|
|
|
|
if (numGenerics != subFunction->generics.size())
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
numGenerics = std::min(superFunction->generics.size(), subFunction->generics.size());
|
2021-12-10 22:05:05 +00:00
|
|
|
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, TypeMismatch{superTy, subTy, "different number of generic type parameters"}});
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
size_t numGenericPacks = superFunction->genericPacks.size();
|
|
|
|
if (numGenericPacks != subFunction->genericPacks.size())
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
numGenericPacks = std::min(superFunction->genericPacks.size(), subFunction->genericPacks.size());
|
2021-12-10 22:05:05 +00:00
|
|
|
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, TypeMismatch{superTy, subTy, "different number of generic type pack parameters"}});
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2021-11-12 14:27:34 +00:00
|
|
|
for (size_t i = 0; i < numGenerics; i++)
|
2022-01-07 01:46:53 +00:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
log.pushSeen(superFunction->generics[i], subFunction->generics[i]);
|
2022-01-07 01:46:53 +00:00
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-04-29 02:24:24 +01:00
|
|
|
for (size_t i = 0; i < numGenericPacks; i++)
|
2022-03-04 16:36:33 +00:00
|
|
|
{
|
2022-04-29 02:24:24 +01:00
|
|
|
log.pushSeen(superFunction->genericPacks[i], subFunction->genericPacks[i]);
|
2022-03-04 16:36:33 +00:00
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
CountMismatch::Context context = ctx;
|
|
|
|
|
|
|
|
if (!isFunctionCall)
|
|
|
|
{
|
|
|
|
Unifier innerState = makeChildUnifier();
|
|
|
|
|
2022-01-27 23:46:05 +00:00
|
|
|
innerState.ctx = CountMismatch::Arg;
|
|
|
|
innerState.tryUnify_(superFunction->argTypes, subFunction->argTypes, isFunctionCall);
|
|
|
|
|
|
|
|
bool reported = !innerState.errors.empty();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-01-27 23:46:05 +00:00
|
|
|
if (auto e = hasUnificationTooComplex(innerState.errors))
|
|
|
|
reportError(*e);
|
|
|
|
else if (!innerState.errors.empty() && innerState.firstPackErrorPos)
|
|
|
|
reportError(
|
|
|
|
TypeError{location, TypeMismatch{superTy, subTy, format("Argument #%d type is not compatible.", *innerState.firstPackErrorPos),
|
|
|
|
innerState.errors.front()}});
|
|
|
|
else if (!innerState.errors.empty())
|
|
|
|
reportError(TypeError{location, TypeMismatch{superTy, subTy, "", innerState.errors.front()}});
|
2021-12-10 22:05:05 +00:00
|
|
|
|
2022-01-27 23:46:05 +00:00
|
|
|
innerState.ctx = CountMismatch::Result;
|
|
|
|
innerState.tryUnify_(subFunction->retType, superFunction->retType);
|
|
|
|
|
|
|
|
if (!reported)
|
|
|
|
{
|
2021-12-10 22:05:05 +00:00
|
|
|
if (auto e = hasUnificationTooComplex(innerState.errors))
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(*e);
|
|
|
|
else if (!innerState.errors.empty() && size(superFunction->retType) == 1 && finite(superFunction->retType))
|
|
|
|
reportError(TypeError{location, TypeMismatch{superTy, subTy, "Return type is not compatible.", innerState.errors.front()}});
|
2021-12-10 22:05:05 +00:00
|
|
|
else if (!innerState.errors.empty() && innerState.firstPackErrorPos)
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(
|
|
|
|
TypeError{location, TypeMismatch{superTy, subTy, format("Return #%d type is not compatible.", *innerState.firstPackErrorPos),
|
2021-12-10 22:05:05 +00:00
|
|
|
innerState.errors.front()}});
|
|
|
|
else if (!innerState.errors.empty())
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, TypeMismatch{superTy, subTy, "", innerState.errors.front()}});
|
2021-12-10 22:05:05 +00:00
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
log.concat(std::move(innerState.log));
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ctx = CountMismatch::Arg;
|
2022-01-07 01:46:53 +00:00
|
|
|
tryUnify_(superFunction->argTypes, subFunction->argTypes, isFunctionCall);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
ctx = CountMismatch::Result;
|
2022-01-07 01:46:53 +00:00
|
|
|
tryUnify_(subFunction->retType, superFunction->retType);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-03-18 00:46:04 +00:00
|
|
|
if (FFlag::LuauTxnLogRefreshFunctionPointers)
|
|
|
|
{
|
|
|
|
// Updating the log may have invalidated the function pointers
|
|
|
|
superFunction = log.getMutable<FunctionTypeVar>(superTy);
|
|
|
|
subFunction = log.getMutable<FunctionTypeVar>(subTy);
|
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
ctx = context;
|
|
|
|
|
2022-04-29 02:24:24 +01:00
|
|
|
for (int i = int(numGenericPacks) - 1; 0 <= i; i--)
|
2022-03-04 16:36:33 +00:00
|
|
|
{
|
2022-04-29 02:24:24 +01:00
|
|
|
log.popSeen(superFunction->genericPacks[i], subFunction->genericPacks[i]);
|
2022-03-04 16:36:33 +00:00
|
|
|
}
|
|
|
|
|
2021-11-12 14:27:34 +00:00
|
|
|
for (int i = int(numGenerics) - 1; 0 <= i; i--)
|
2022-01-07 01:46:53 +00:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
log.popSeen(superFunction->generics[i], subFunction->generics[i]);
|
2022-01-07 01:46:53 +00:00
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
|
|
|
|
struct Resetter
|
|
|
|
{
|
|
|
|
explicit Resetter(Variance* variance)
|
|
|
|
: oldValue(*variance)
|
|
|
|
, variance(variance)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
Variance oldValue;
|
|
|
|
Variance* variance;
|
|
|
|
|
|
|
|
~Resetter()
|
|
|
|
{
|
|
|
|
*variance = oldValue;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
void Unifier::tryUnifyTables(TypeId subTy, TypeId superTy, bool isIntersection)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2021-12-10 22:05:05 +00:00
|
|
|
if (!FFlag::LuauTableSubtypingVariance2)
|
2022-01-07 01:46:53 +00:00
|
|
|
return DEPRECATED_tryUnifyTables(subTy, superTy, isIntersection);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
TableTypeVar* superTable = log.getMutable<TableTypeVar>(superTy);
|
|
|
|
TableTypeVar* subTable = log.getMutable<TableTypeVar>(subTy);
|
2022-02-04 16:45:57 +00:00
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
if (!superTable || !subTable)
|
2021-11-05 02:34:35 +00:00
|
|
|
ice("passed non-table types to unifyTables");
|
|
|
|
|
|
|
|
std::vector<std::string> missingProperties;
|
|
|
|
std::vector<std::string> extraProperties;
|
|
|
|
|
2021-11-05 15:47:21 +00:00
|
|
|
// Optimization: First test that the property sets are compatible without doing any recursive unification
|
2022-03-04 16:36:33 +00:00
|
|
|
if (!subTable->indexer && subTable->state != TableState::Free)
|
2021-11-05 15:47:21 +00:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
for (const auto& [propName, superProp] : superTable->props)
|
2021-11-05 15:47:21 +00:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
auto subIter = subTable->props.find(propName);
|
2022-02-04 16:45:57 +00:00
|
|
|
|
2022-05-20 01:02:24 +01:00
|
|
|
if (subIter == subTable->props.end() && (!FFlag::LuauSubtypingAddOptPropsToUnsealedTables || subTable->state == TableState::Unsealed) &&
|
|
|
|
!isOptional(superProp.type))
|
|
|
|
missingProperties.push_back(propName);
|
2021-11-05 15:47:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!missingProperties.empty())
|
|
|
|
{
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, MissingProperties{superTy, subTy, std::move(missingProperties)}});
|
2021-11-05 15:47:21 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// And vice versa if we're invariant
|
2022-04-15 00:57:43 +01:00
|
|
|
if (variance == Invariant && !superTable->indexer && superTable->state != TableState::Unsealed && superTable->state != TableState::Free)
|
2021-11-05 15:47:21 +00:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
for (const auto& [propName, subProp] : subTable->props)
|
2021-11-05 15:47:21 +00:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
auto superIter = superTable->props.find(propName);
|
2022-02-04 16:45:57 +00:00
|
|
|
|
2022-05-20 01:02:24 +01:00
|
|
|
if (superIter == superTable->props.end() && (FFlag::LuauSubtypingAddOptPropsToUnsealedTables || !isOptional(subProp.type)))
|
|
|
|
extraProperties.push_back(propName);
|
2021-11-05 15:47:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!extraProperties.empty())
|
|
|
|
{
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, MissingProperties{superTy, subTy, std::move(extraProperties), MissingProperties::Extra}});
|
2021-11-05 15:47:21 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2021-11-19 16:10:07 +00:00
|
|
|
|
2021-11-05 02:34:35 +00:00
|
|
|
// Width subtyping: any property in the supertype must be in the subtype,
|
|
|
|
// and the types must agree.
|
2022-01-07 01:46:53 +00:00
|
|
|
for (const auto& [name, prop] : superTable->props)
|
2021-11-05 02:34:35 +00:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
const auto& r = subTable->props.find(name);
|
|
|
|
if (r != subTable->props.end())
|
2021-11-05 02:34:35 +00:00
|
|
|
{
|
|
|
|
// TODO: read-only properties don't need invariance
|
|
|
|
Resetter resetter{&variance};
|
|
|
|
variance = Invariant;
|
|
|
|
|
|
|
|
Unifier innerState = makeChildUnifier();
|
2022-01-07 01:46:53 +00:00
|
|
|
innerState.tryUnify_(r->second.type, prop.type);
|
2021-11-12 14:27:34 +00:00
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
checkChildUnifierTypeMismatch(innerState.errors, name, superTy, subTy);
|
2021-11-12 14:27:34 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (innerState.errors.empty())
|
|
|
|
log.concat(std::move(innerState.log));
|
2021-11-05 02:34:35 +00:00
|
|
|
}
|
2022-03-04 16:36:33 +00:00
|
|
|
else if (subTable->indexer && maybeString(subTable->indexer->indexType))
|
2021-11-05 02:34:35 +00:00
|
|
|
{
|
|
|
|
// TODO: read-only indexers don't need invariance
|
|
|
|
// TODO: really we should only allow this if prop.type is optional.
|
|
|
|
Resetter resetter{&variance};
|
|
|
|
variance = Invariant;
|
|
|
|
|
|
|
|
Unifier innerState = makeChildUnifier();
|
2022-01-07 01:46:53 +00:00
|
|
|
innerState.tryUnify_(subTable->indexer->indexResultType, prop.type);
|
2021-11-12 14:27:34 +00:00
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
checkChildUnifierTypeMismatch(innerState.errors, name, superTy, subTy);
|
2021-11-12 14:27:34 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (innerState.errors.empty())
|
|
|
|
log.concat(std::move(innerState.log));
|
2021-11-05 02:34:35 +00:00
|
|
|
}
|
2022-05-20 01:02:24 +01:00
|
|
|
else if ((!FFlag::LuauSubtypingAddOptPropsToUnsealedTables || subTable->state == TableState::Unsealed) && isOptional(prop.type))
|
2022-03-04 16:36:33 +00:00
|
|
|
// This is sound because unsealed table types are precise, so `{ p : T } <: { p : T, q : U? }`
|
|
|
|
// since if `t : { p : T }` then we are guaranteed that `t.q` is `nil`.
|
|
|
|
// TODO: if the supertype is written to, the subtype may no longer be precise (alias analysis?)
|
2021-11-05 15:47:21 +00:00
|
|
|
{
|
|
|
|
}
|
2022-01-07 01:46:53 +00:00
|
|
|
else if (subTable->state == TableState::Free)
|
2021-11-05 02:34:35 +00:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
PendingType* pendingSub = log.queue(subTy);
|
|
|
|
TableTypeVar* ttv = getMutable<TableTypeVar>(pendingSub);
|
|
|
|
LUAU_ASSERT(ttv);
|
|
|
|
ttv->props[name] = prop;
|
|
|
|
subTable = ttv;
|
2021-11-05 02:34:35 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
missingProperties.push_back(name);
|
2022-03-11 16:55:02 +00:00
|
|
|
|
2022-04-29 02:24:24 +01:00
|
|
|
// Recursive unification can change the txn log, and invalidate the old
|
|
|
|
// table. If we detect that this has happened, we start over, with the updated
|
|
|
|
// txn log.
|
|
|
|
TableTypeVar* newSuperTable = log.getMutable<TableTypeVar>(superTy);
|
|
|
|
TableTypeVar* newSubTable = log.getMutable<TableTypeVar>(subTy);
|
|
|
|
if (superTable != newSuperTable || subTable != newSubTable)
|
2022-03-11 16:55:02 +00:00
|
|
|
{
|
2022-04-29 02:24:24 +01:00
|
|
|
if (errors.empty())
|
|
|
|
return tryUnifyTables(subTy, superTy, isIntersection);
|
|
|
|
else
|
|
|
|
return;
|
2022-03-11 16:55:02 +00:00
|
|
|
}
|
2021-11-05 02:34:35 +00:00
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
for (const auto& [name, prop] : subTable->props)
|
2021-11-05 02:34:35 +00:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
if (superTable->props.count(name))
|
2021-11-05 02:34:35 +00:00
|
|
|
{
|
|
|
|
// If both lt and rt contain the property, then
|
|
|
|
// we're done since we already unified them above
|
|
|
|
}
|
2022-03-04 16:36:33 +00:00
|
|
|
else if (superTable->indexer && maybeString(superTable->indexer->indexType))
|
2021-11-05 02:34:35 +00:00
|
|
|
{
|
|
|
|
// TODO: read-only indexers don't need invariance
|
|
|
|
// TODO: really we should only allow this if prop.type is optional.
|
|
|
|
Resetter resetter{&variance};
|
|
|
|
variance = Invariant;
|
|
|
|
|
|
|
|
Unifier innerState = makeChildUnifier();
|
2022-01-07 01:46:53 +00:00
|
|
|
innerState.tryUnify_(superTable->indexer->indexResultType, prop.type);
|
2021-11-12 14:27:34 +00:00
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
checkChildUnifierTypeMismatch(innerState.errors, name, superTy, subTy);
|
2021-11-12 14:27:34 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (innerState.errors.empty())
|
|
|
|
log.concat(std::move(innerState.log));
|
2021-11-05 02:34:35 +00:00
|
|
|
}
|
2022-01-07 01:46:53 +00:00
|
|
|
else if (superTable->state == TableState::Unsealed)
|
2021-11-05 02:34:35 +00:00
|
|
|
{
|
|
|
|
// TODO: this case is unsound when variance is Invariant, but without it lua-apps fails to typecheck.
|
|
|
|
// TODO: file a JIRA
|
|
|
|
// TODO: hopefully readonly/writeonly properties will fix this.
|
|
|
|
Property clone = prop;
|
|
|
|
clone.type = deeplyOptional(clone.type);
|
2022-01-07 01:46:53 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
PendingType* pendingSuper = log.queue(superTy);
|
|
|
|
TableTypeVar* pendingSuperTtv = getMutable<TableTypeVar>(pendingSuper);
|
|
|
|
pendingSuperTtv->props[name] = clone;
|
|
|
|
superTable = pendingSuperTtv;
|
2021-11-05 02:34:35 +00:00
|
|
|
}
|
|
|
|
else if (variance == Covariant)
|
2021-11-05 15:47:21 +00:00
|
|
|
{
|
|
|
|
}
|
2022-05-20 01:02:24 +01:00
|
|
|
else if (!FFlag::LuauSubtypingAddOptPropsToUnsealedTables && isOptional(prop.type))
|
2021-11-05 15:47:21 +00:00
|
|
|
{
|
|
|
|
}
|
2022-01-07 01:46:53 +00:00
|
|
|
else if (superTable->state == TableState::Free)
|
2021-11-05 02:34:35 +00:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
PendingType* pendingSuper = log.queue(superTy);
|
|
|
|
TableTypeVar* pendingSuperTtv = getMutable<TableTypeVar>(pendingSuper);
|
|
|
|
pendingSuperTtv->props[name] = prop;
|
|
|
|
superTable = pendingSuperTtv;
|
2021-11-05 02:34:35 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
extraProperties.push_back(name);
|
2022-03-11 16:55:02 +00:00
|
|
|
|
2022-04-29 02:24:24 +01:00
|
|
|
// Recursive unification can change the txn log, and invalidate the old
|
|
|
|
// table. If we detect that this has happened, we start over, with the updated
|
|
|
|
// txn log.
|
|
|
|
TableTypeVar* newSuperTable = log.getMutable<TableTypeVar>(superTy);
|
|
|
|
TableTypeVar* newSubTable = log.getMutable<TableTypeVar>(subTy);
|
|
|
|
if (superTable != newSuperTable || subTable != newSubTable)
|
2022-03-11 16:55:02 +00:00
|
|
|
{
|
2022-04-29 02:24:24 +01:00
|
|
|
if (errors.empty())
|
|
|
|
return tryUnifyTables(subTy, superTy, isIntersection);
|
|
|
|
else
|
|
|
|
return;
|
2022-03-11 16:55:02 +00:00
|
|
|
}
|
2021-11-05 02:34:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Unify indexers
|
2022-01-07 01:46:53 +00:00
|
|
|
if (superTable->indexer && subTable->indexer)
|
2021-11-05 02:34:35 +00:00
|
|
|
{
|
|
|
|
// TODO: read-only indexers don't need invariance
|
|
|
|
Resetter resetter{&variance};
|
|
|
|
variance = Invariant;
|
|
|
|
|
|
|
|
Unifier innerState = makeChildUnifier();
|
2022-03-24 22:04:14 +00:00
|
|
|
|
2022-04-21 22:44:27 +01:00
|
|
|
innerState.tryUnify_(subTable->indexer->indexType, superTable->indexer->indexType);
|
2022-03-24 22:04:14 +00:00
|
|
|
|
2022-04-21 22:44:27 +01:00
|
|
|
bool reported = !innerState.errors.empty();
|
2022-03-24 22:04:14 +00:00
|
|
|
|
2022-04-21 22:44:27 +01:00
|
|
|
checkChildUnifierTypeMismatch(innerState.errors, "[indexer key]", superTy, subTy);
|
2022-03-24 22:04:14 +00:00
|
|
|
|
2022-04-21 22:44:27 +01:00
|
|
|
innerState.tryUnify_(subTable->indexer->indexResultType, superTable->indexer->indexResultType);
|
2022-03-24 22:04:14 +00:00
|
|
|
|
2022-04-21 22:44:27 +01:00
|
|
|
if (!reported)
|
|
|
|
checkChildUnifierTypeMismatch(innerState.errors, "[indexer value]", superTy, subTy);
|
2022-01-07 01:46:53 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (innerState.errors.empty())
|
|
|
|
log.concat(std::move(innerState.log));
|
2021-11-05 02:34:35 +00:00
|
|
|
}
|
2022-01-07 01:46:53 +00:00
|
|
|
else if (superTable->indexer)
|
2021-11-05 02:34:35 +00:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
if (subTable->state == TableState::Unsealed || subTable->state == TableState::Free)
|
2021-11-05 02:34:35 +00:00
|
|
|
{
|
|
|
|
// passing/assigning a table without an indexer to something that has one
|
|
|
|
// e.g. table.insert(t, 1) where t is a non-sealed table and doesn't have an indexer.
|
|
|
|
// TODO: we only need to do this if the supertype's indexer is read/write
|
|
|
|
// since that can add indexed elements.
|
2022-03-11 16:55:02 +00:00
|
|
|
log.changeIndexer(subTy, superTable->indexer);
|
2021-11-05 02:34:35 +00:00
|
|
|
}
|
|
|
|
}
|
2022-01-07 01:46:53 +00:00
|
|
|
else if (subTable->indexer && variance == Invariant)
|
2021-11-05 02:34:35 +00:00
|
|
|
{
|
|
|
|
// Symmetric if we are invariant
|
2022-01-07 01:46:53 +00:00
|
|
|
if (superTable->state == TableState::Unsealed || superTable->state == TableState::Free)
|
2021-11-05 02:34:35 +00:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
log.changeIndexer(superTy, subTable->indexer);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-29 02:24:24 +01:00
|
|
|
// Changing the indexer can invalidate the table pointers.
|
|
|
|
superTable = log.getMutable<TableTypeVar>(superTy);
|
|
|
|
subTable = log.getMutable<TableTypeVar>(subTy);
|
2021-11-05 02:34:35 +00:00
|
|
|
|
|
|
|
if (!missingProperties.empty())
|
|
|
|
{
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, MissingProperties{superTy, subTy, std::move(missingProperties)}});
|
2021-11-05 02:34:35 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!extraProperties.empty())
|
|
|
|
{
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, MissingProperties{superTy, subTy, std::move(extraProperties), MissingProperties::Extra}});
|
2021-11-05 02:34:35 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* TypeVars are commonly cyclic, so it is entirely possible
|
|
|
|
* for unifying a property of a table to change the table itself!
|
|
|
|
* We need to check for this and start over if we notice this occurring.
|
|
|
|
*
|
|
|
|
* I believe this is guaranteed to terminate eventually because this will
|
|
|
|
* only happen when a free table is bound to another table.
|
|
|
|
*/
|
2022-01-07 01:46:53 +00:00
|
|
|
if (superTable->boundTo || subTable->boundTo)
|
|
|
|
return tryUnify_(subTy, superTy);
|
2021-11-05 02:34:35 +00:00
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
if (superTable->state == TableState::Free)
|
2021-11-05 02:34:35 +00:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
log.bindTable(superTy, subTy);
|
2021-11-05 02:34:35 +00:00
|
|
|
}
|
2022-01-07 01:46:53 +00:00
|
|
|
else if (subTable->state == TableState::Free)
|
2021-11-05 02:34:35 +00:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
log.bindTable(subTy, superTy);
|
2021-11-05 02:34:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TypeId Unifier::deeplyOptional(TypeId ty, std::unordered_map<TypeId, TypeId> seen)
|
|
|
|
{
|
|
|
|
ty = follow(ty);
|
2022-05-20 01:02:24 +01:00
|
|
|
if (isOptional(ty))
|
2021-11-05 02:34:35 +00:00
|
|
|
return ty;
|
|
|
|
else if (const TableTypeVar* ttv = get<TableTypeVar>(ty))
|
|
|
|
{
|
|
|
|
TypeId& result = seen[ty];
|
|
|
|
if (result)
|
|
|
|
return result;
|
|
|
|
result = types->addType(*ttv);
|
|
|
|
TableTypeVar* resultTtv = getMutable<TableTypeVar>(result);
|
|
|
|
for (auto& [name, prop] : resultTtv->props)
|
|
|
|
prop.type = deeplyOptional(prop.type, seen);
|
2021-12-10 22:05:05 +00:00
|
|
|
return types->addType(UnionTypeVar{{getSingletonTypes().nilType, result}});
|
2021-11-05 02:34:35 +00:00
|
|
|
}
|
|
|
|
else
|
2021-12-10 22:05:05 +00:00
|
|
|
return types->addType(UnionTypeVar{{getSingletonTypes().nilType, ty}});
|
2021-11-05 02:34:35 +00:00
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
void Unifier::DEPRECATED_tryUnifyTables(TypeId subTy, TypeId superTy, bool isIntersection)
|
2021-11-05 02:34:35 +00:00
|
|
|
{
|
2021-12-10 22:05:05 +00:00
|
|
|
LUAU_ASSERT(!FFlag::LuauTableSubtypingVariance2);
|
2021-11-05 02:34:35 +00:00
|
|
|
Resetter resetter{&variance};
|
2021-10-29 21:25:12 +01:00
|
|
|
variance = Invariant;
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
TableTypeVar* superTable = log.getMutable<TableTypeVar>(superTy);
|
|
|
|
TableTypeVar* subTable = log.getMutable<TableTypeVar>(subTy);
|
2022-01-07 01:46:53 +00:00
|
|
|
|
|
|
|
if (!superTable || !subTable)
|
2021-10-29 21:25:12 +01:00
|
|
|
ice("passed non-table types to unifyTables");
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
if (superTable->state == TableState::Sealed && subTable->state == TableState::Sealed)
|
|
|
|
return tryUnifySealedTables(subTy, superTy, isIntersection);
|
|
|
|
else if ((superTable->state == TableState::Sealed && subTable->state == TableState::Unsealed) ||
|
|
|
|
(superTable->state == TableState::Unsealed && subTable->state == TableState::Sealed))
|
|
|
|
return tryUnifySealedTables(subTy, superTy, isIntersection);
|
|
|
|
else if ((superTable->state == TableState::Sealed && subTable->state == TableState::Generic) ||
|
|
|
|
(superTable->state == TableState::Generic && subTable->state == TableState::Sealed))
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, TypeMismatch{superTy, subTy}});
|
2022-01-07 01:46:53 +00:00
|
|
|
else if ((superTable->state == TableState::Free) != (subTable->state == TableState::Free)) // one table is free and the other is not
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
TypeId freeTypeId = subTable->state == TableState::Free ? subTy : superTy;
|
|
|
|
TypeId otherTypeId = subTable->state == TableState::Free ? superTy : subTy;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
return tryUnifyFreeTable(otherTypeId, freeTypeId);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-01-07 01:46:53 +00:00
|
|
|
else if (superTable->state == TableState::Free && subTable->state == TableState::Free)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
tryUnifyFreeTable(subTy, superTy);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
// avoid creating a cycle when the types are already pointing at each other
|
2022-01-07 01:46:53 +00:00
|
|
|
if (follow(superTy) != follow(subTy))
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
log.bindTable(superTy, subTy);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2022-01-07 01:46:53 +00:00
|
|
|
else if (superTable->state != TableState::Sealed && subTable->state != TableState::Sealed)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
// All free tables are checked in one of the branches above
|
2022-01-07 01:46:53 +00:00
|
|
|
LUAU_ASSERT(superTable->state != TableState::Free);
|
|
|
|
LUAU_ASSERT(subTable->state != TableState::Free);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
// Tables must have exactly the same props and their types must all unify
|
|
|
|
// I honestly have no idea if this is remotely close to reasonable.
|
2022-01-07 01:46:53 +00:00
|
|
|
for (const auto& [name, prop] : superTable->props)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
const auto& r = subTable->props.find(name);
|
|
|
|
if (r == subTable->props.end())
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, UnknownProperty{subTy, name}});
|
2021-10-29 21:25:12 +01:00
|
|
|
else
|
2022-01-07 01:46:53 +00:00
|
|
|
tryUnify_(r->second.type, prop.type);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
if (superTable->indexer && subTable->indexer)
|
|
|
|
tryUnifyIndexer(*subTable->indexer, *superTable->indexer);
|
|
|
|
else if (superTable->indexer)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
// passing/assigning a table without an indexer to something that has one
|
|
|
|
// e.g. table.insert(t, 1) where t is a non-sealed table and doesn't have an indexer.
|
2022-01-07 01:46:53 +00:00
|
|
|
if (subTable->state == TableState::Unsealed)
|
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
log.changeIndexer(subTy, superTable->indexer);
|
2022-01-07 01:46:53 +00:00
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
else
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, CannotExtendTable{subTy, CannotExtendTable::Indexer}});
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
2022-01-07 01:46:53 +00:00
|
|
|
else if (superTable->state == TableState::Sealed)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
// lt is sealed and so it must be possible for rt to have precisely the same shape
|
|
|
|
// Verify that this is the case, then bind rt to lt.
|
|
|
|
ice("unsealed tables are not working yet", location);
|
|
|
|
}
|
2022-01-07 01:46:53 +00:00
|
|
|
else if (subTable->state == TableState::Sealed)
|
|
|
|
return tryUnifyTables(superTy, subTy, isIntersection);
|
2021-10-29 21:25:12 +01:00
|
|
|
else
|
|
|
|
ice("tryUnifyTables");
|
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
void Unifier::tryUnifyFreeTable(TypeId subTy, TypeId superTy)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-24 22:04:14 +00:00
|
|
|
LUAU_ASSERT(!FFlag::LuauTableSubtypingVariance2);
|
2022-03-11 16:55:02 +00:00
|
|
|
TableTypeVar* freeTable = log.getMutable<TableTypeVar>(superTy);
|
|
|
|
TableTypeVar* subTable = log.getMutable<TableTypeVar>(subTy);
|
2022-01-07 01:46:53 +00:00
|
|
|
|
|
|
|
if (!freeTable || !subTable)
|
2021-10-29 21:25:12 +01:00
|
|
|
ice("passed non-table types to tryUnifyFreeTable");
|
|
|
|
|
|
|
|
// Any properties in freeTable must unify with those in otherTable.
|
|
|
|
// Then bind freeTable to otherTable.
|
|
|
|
for (const auto& [freeName, freeProp] : freeTable->props)
|
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
if (auto subProp = findTablePropertyRespectingMeta(subTy, freeName))
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-05-26 23:08:16 +01:00
|
|
|
tryUnify_(*subProp, freeProp.type);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* TypeVars are commonly cyclic, so it is entirely possible
|
|
|
|
* for unifying a property of a table to change the table itself!
|
|
|
|
* We need to check for this and start over if we notice this occurring.
|
|
|
|
*
|
|
|
|
* I believe this is guaranteed to terminate eventually because this will
|
|
|
|
* only happen when a free table is bound to another table.
|
|
|
|
*/
|
2022-03-11 16:55:02 +00:00
|
|
|
if (!log.getMutable<TableTypeVar>(superTy) || !log.getMutable<TableTypeVar>(subTy))
|
|
|
|
return tryUnify_(subTy, superTy);
|
2022-01-07 01:46:53 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (TableTypeVar* pendingFreeTtv = log.getMutable<TableTypeVar>(superTy); pendingFreeTtv && pendingFreeTtv->boundTo)
|
|
|
|
return tryUnify_(subTy, superTy);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// If the other table is also free, then we are learning that it has more
|
|
|
|
// properties than we previously thought. Else, it is an error.
|
2022-01-07 01:46:53 +00:00
|
|
|
if (subTable->state == TableState::Free)
|
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
PendingType* pendingSub = log.queue(subTy);
|
|
|
|
TableTypeVar* pendingSubTtv = getMutable<TableTypeVar>(pendingSub);
|
|
|
|
LUAU_ASSERT(pendingSubTtv);
|
|
|
|
pendingSubTtv->props.insert({freeName, freeProp});
|
2022-01-07 01:46:53 +00:00
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
else
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, UnknownProperty{subTy, freeName}});
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
if (freeTable->indexer && subTable->indexer)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
Unifier innerState = makeChildUnifier();
|
2022-01-07 01:46:53 +00:00
|
|
|
innerState.tryUnifyIndexer(*subTable->indexer, *freeTable->indexer);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
checkChildUnifierTypeMismatch(innerState.errors, superTy, subTy);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
log.concat(std::move(innerState.log));
|
2022-01-07 01:46:53 +00:00
|
|
|
}
|
|
|
|
else if (subTable->state == TableState::Free && freeTable->indexer)
|
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
log.changeIndexer(superTy, subTable->indexer);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
if (!freeTable->boundTo && subTable->state != TableState::Free)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
log.bindTable(superTy, subTy);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
void Unifier::tryUnifySealedTables(TypeId subTy, TypeId superTy, bool isIntersection)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-24 22:04:14 +00:00
|
|
|
LUAU_ASSERT(!FFlag::LuauTableSubtypingVariance2);
|
2022-03-11 16:55:02 +00:00
|
|
|
TableTypeVar* superTable = log.getMutable<TableTypeVar>(superTy);
|
|
|
|
TableTypeVar* subTable = log.getMutable<TableTypeVar>(subTy);
|
2022-01-07 01:46:53 +00:00
|
|
|
|
|
|
|
if (!superTable || !subTable)
|
2021-10-29 21:25:12 +01:00
|
|
|
ice("passed non-table types to unifySealedTables");
|
|
|
|
|
|
|
|
std::vector<std::string> missingPropertiesInSuper;
|
2022-01-07 01:46:53 +00:00
|
|
|
bool isUnnamedTable = subTable->name == std::nullopt && subTable->syntheticName == std::nullopt;
|
2021-10-29 21:25:12 +01:00
|
|
|
bool errorReported = false;
|
|
|
|
|
|
|
|
// Optimization: First test that the property sets are compatible without doing any recursive unification
|
2022-03-04 16:36:33 +00:00
|
|
|
if (!subTable->indexer)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
for (const auto& [propName, superProp] : superTable->props)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
auto subIter = subTable->props.find(propName);
|
|
|
|
if (subIter == subTable->props.end() && !isOptional(superProp.type))
|
2021-10-29 21:25:12 +01:00
|
|
|
missingPropertiesInSuper.push_back(propName);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!missingPropertiesInSuper.empty())
|
|
|
|
{
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, MissingProperties{superTy, subTy, std::move(missingPropertiesInSuper)}});
|
2021-10-29 21:25:12 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-15 00:57:43 +01:00
|
|
|
Unifier innerState = makeChildUnifier();
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
// Tables must have exactly the same props and their types must all unify
|
2022-01-07 01:46:53 +00:00
|
|
|
for (const auto& it : superTable->props)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
const auto& r = subTable->props.find(it.first);
|
|
|
|
if (r == subTable->props.end())
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2021-11-12 14:27:34 +00:00
|
|
|
if (isOptional(it.second.type))
|
|
|
|
continue;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
missingPropertiesInSuper.push_back(it.first);
|
|
|
|
|
2022-01-27 23:46:05 +00:00
|
|
|
innerState.reportError(TypeError{location, TypeMismatch{superTy, subTy}});
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (isUnnamedTable && r->second.location)
|
|
|
|
{
|
|
|
|
size_t oldErrorSize = innerState.errors.size();
|
|
|
|
Location old = innerState.location;
|
|
|
|
innerState.location = *r->second.location;
|
2022-01-07 01:46:53 +00:00
|
|
|
innerState.tryUnify_(r->second.type, it.second.type);
|
2021-10-29 21:25:12 +01:00
|
|
|
innerState.location = old;
|
|
|
|
|
|
|
|
if (oldErrorSize != innerState.errors.size() && !errorReported)
|
|
|
|
{
|
|
|
|
errorReported = true;
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(innerState.errors.back());
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
innerState.tryUnify_(r->second.type, it.second.type);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
if (superTable->indexer || subTable->indexer)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
if (superTable->indexer && subTable->indexer)
|
|
|
|
innerState.tryUnifyIndexer(*subTable->indexer, *superTable->indexer);
|
|
|
|
else if (subTable->state == TableState::Unsealed)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
if (superTable->indexer && !subTable->indexer)
|
2022-01-07 01:46:53 +00:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
log.changeIndexer(subTy, superTable->indexer);
|
2022-01-07 01:46:53 +00:00
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-03-11 16:55:02 +00:00
|
|
|
else if (superTable->state == TableState::Unsealed)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
if (subTable->indexer && !superTable->indexer)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
log.changeIndexer(superTy, subTable->indexer);
|
2022-01-07 01:46:53 +00:00
|
|
|
}
|
2022-03-11 16:55:02 +00:00
|
|
|
}
|
|
|
|
else if (superTable->indexer)
|
|
|
|
{
|
|
|
|
innerState.tryUnify_(getSingletonTypes().stringType, superTable->indexer->indexType);
|
|
|
|
for (const auto& [name, type] : subTable->props)
|
2022-01-07 01:46:53 +00:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
const auto& it = superTable->props.find(name);
|
|
|
|
if (it == superTable->props.end())
|
|
|
|
innerState.tryUnify_(type.type, superTable->indexer->indexResultType);
|
2022-01-07 01:46:53 +00:00
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-03-11 16:55:02 +00:00
|
|
|
else
|
|
|
|
innerState.reportError(TypeError{location, TypeMismatch{superTy, subTy}});
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (!errorReported)
|
|
|
|
log.concat(std::move(innerState.log));
|
2022-01-07 01:46:53 +00:00
|
|
|
else
|
2021-10-29 21:25:12 +01:00
|
|
|
return;
|
|
|
|
|
|
|
|
if (!missingPropertiesInSuper.empty())
|
|
|
|
{
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, MissingProperties{superTy, subTy, std::move(missingPropertiesInSuper)}});
|
2021-10-29 21:25:12 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
// If the superTy is an immediate part of an intersection type, do not do extra-property check.
|
2021-10-29 21:25:12 +01:00
|
|
|
// Otherwise, we would falsely generate an extra-property-error for 's' in this code:
|
|
|
|
// local a: {n: number} & {s: string} = {n=1, s=""}
|
2021-11-04 14:50:46 +00:00
|
|
|
// When checking against the table '{n: number}'.
|
2022-01-07 01:46:53 +00:00
|
|
|
if (!isIntersection && superTable->state != TableState::Unsealed && !superTable->indexer)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
// Check for extra properties in the subTy
|
|
|
|
std::vector<std::string> extraPropertiesInSub;
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
for (const auto& [subKey, subProp] : subTable->props)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
const auto& superIt = superTable->props.find(subKey);
|
|
|
|
if (superIt == superTable->props.end())
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
if (isOptional(subProp.type))
|
2021-11-12 14:27:34 +00:00
|
|
|
continue;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
extraPropertiesInSub.push_back(subKey);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!extraPropertiesInSub.empty())
|
|
|
|
{
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, MissingProperties{superTy, subTy, std::move(extraPropertiesInSub), MissingProperties::Extra}});
|
2021-10-29 21:25:12 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
checkChildUnifierTypeMismatch(innerState.errors, superTy, subTy);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
void Unifier::tryUnifyWithMetatable(TypeId subTy, TypeId superTy, bool reversed)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
const MetatableTypeVar* superMetatable = get<MetatableTypeVar>(superTy);
|
|
|
|
if (!superMetatable)
|
2021-10-29 21:25:12 +01:00
|
|
|
ice("tryUnifyMetatable invoked with non-metatable TypeVar");
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
TypeError mismatchError = TypeError{location, TypeMismatch{reversed ? subTy : superTy, reversed ? superTy : subTy}};
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (const MetatableTypeVar* subMetatable = log.getMutable<MetatableTypeVar>(subTy))
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
Unifier innerState = makeChildUnifier();
|
2022-01-07 01:46:53 +00:00
|
|
|
innerState.tryUnify_(subMetatable->table, superMetatable->table);
|
|
|
|
innerState.tryUnify_(subMetatable->metatable, superMetatable->metatable);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-01-06 23:26:14 +00:00
|
|
|
if (auto e = hasUnificationTooComplex(innerState.errors))
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(*e);
|
2022-01-06 23:26:14 +00:00
|
|
|
else if (!innerState.errors.empty())
|
2022-02-04 16:45:57 +00:00
|
|
|
reportError(TypeError{location, TypeMismatch{reversed ? subTy : superTy, reversed ? superTy : subTy, "", innerState.errors.front()}});
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
log.concat(std::move(innerState.log));
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-03-11 16:55:02 +00:00
|
|
|
else if (TableTypeVar* subTable = log.getMutable<TableTypeVar>(subTy))
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
switch (subTable->state)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
case TableState::Free:
|
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
tryUnify_(subTy, superMetatable->table);
|
2022-03-11 16:55:02 +00:00
|
|
|
log.bindTable(subTy, superTy);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// We know the shape of sealed, unsealed, and generic tables; you can't add a metatable on to any of these.
|
|
|
|
case TableState::Sealed:
|
|
|
|
case TableState::Unsealed:
|
|
|
|
case TableState::Generic:
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(mismatchError);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
2022-03-11 16:55:02 +00:00
|
|
|
else if (log.getMutable<AnyTypeVar>(subTy) || log.getMutable<ErrorTypeVar>(subTy))
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(mismatchError);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Class unification is almost, but not quite symmetrical. We use the 'reversed' boolean to indicate which scenario we are evaluating.
|
2022-01-07 01:46:53 +00:00
|
|
|
void Unifier::tryUnifyWithClass(TypeId subTy, TypeId superTy, bool reversed)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
if (reversed)
|
|
|
|
std::swap(superTy, subTy);
|
|
|
|
|
|
|
|
auto fail = [&]() {
|
|
|
|
if (!reversed)
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, TypeMismatch{superTy, subTy}});
|
2021-10-29 21:25:12 +01:00
|
|
|
else
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, TypeMismatch{subTy, superTy}});
|
2021-10-29 21:25:12 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
const ClassTypeVar* superClass = get<ClassTypeVar>(superTy);
|
|
|
|
if (!superClass)
|
|
|
|
ice("tryUnifyClass invoked with non-class TypeVar");
|
|
|
|
|
|
|
|
if (const ClassTypeVar* subClass = get<ClassTypeVar>(subTy))
|
|
|
|
{
|
|
|
|
switch (variance)
|
|
|
|
{
|
|
|
|
case Covariant:
|
|
|
|
if (!isSubclass(subClass, superClass))
|
|
|
|
return fail();
|
|
|
|
return;
|
|
|
|
case Invariant:
|
|
|
|
if (subClass != superClass)
|
|
|
|
return fail();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
ice("Illegal variance setting!");
|
|
|
|
}
|
2022-01-07 01:46:53 +00:00
|
|
|
else if (TableTypeVar* subTable = getMutable<TableTypeVar>(subTy))
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
/**
|
|
|
|
* A free table is something whose shape we do not exactly know yet.
|
|
|
|
* Thus, it is entirely reasonable that we might discover that it is being used as some class type.
|
|
|
|
* In this case, the free table must indeed be that exact class.
|
|
|
|
* For this to hold, the table must not have any properties that the class does not.
|
|
|
|
* Further, all properties of the table should unify cleanly with the matching class properties.
|
|
|
|
* TODO: What does it mean for the table to have an indexer? (probably failure?)
|
|
|
|
*
|
|
|
|
* Tables that are not free are known to be actual tables.
|
|
|
|
*/
|
2022-01-07 01:46:53 +00:00
|
|
|
if (subTable->state != TableState::Free)
|
2021-10-29 21:25:12 +01:00
|
|
|
return fail();
|
|
|
|
|
|
|
|
bool ok = true;
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
for (const auto& [propName, prop] : subTable->props)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
const Property* classProp = lookupClassProp(superClass, propName);
|
|
|
|
if (!classProp)
|
|
|
|
{
|
|
|
|
ok = false;
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, UnknownProperty{superTy, propName}});
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else
|
2021-11-12 14:27:34 +00:00
|
|
|
{
|
2022-01-06 23:26:14 +00:00
|
|
|
Unifier innerState = makeChildUnifier();
|
2022-01-07 01:46:53 +00:00
|
|
|
innerState.tryUnify_(classProp->type, prop.type);
|
2021-11-12 14:27:34 +00:00
|
|
|
|
2022-01-06 23:26:14 +00:00
|
|
|
checkChildUnifierTypeMismatch(innerState.errors, propName, reversed ? subTy : superTy, reversed ? superTy : subTy);
|
2021-11-12 14:27:34 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (innerState.errors.empty())
|
2022-01-06 23:26:14 +00:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
log.concat(std::move(innerState.log));
|
2021-11-12 14:27:34 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
ok = false;
|
2021-11-12 14:27:34 +00:00
|
|
|
}
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
if (subTable->indexer)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
ok = false;
|
|
|
|
std::string msg = "Class " + superClass->name + " does not have an indexer";
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, GenericError{msg}});
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!ok)
|
|
|
|
return;
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
log.bindTable(subTy, superTy);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
return fail();
|
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
void Unifier::tryUnifyIndexer(const TableIndexer& subIndexer, const TableIndexer& superIndexer)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-04-21 22:44:27 +01:00
|
|
|
LUAU_ASSERT(!FFlag::LuauTableSubtypingVariance2);
|
2022-03-24 22:04:14 +00:00
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
tryUnify_(subIndexer.indexType, superIndexer.indexType);
|
|
|
|
tryUnify_(subIndexer.indexResultType, superIndexer.indexResultType);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2021-11-05 02:34:35 +00:00
|
|
|
static void queueTypePack(std::vector<TypeId>& queue, DenseHashSet<TypePackId>& seenTypePacks, Unifier& state, TypePackId a, TypePackId anyTypePack)
|
|
|
|
{
|
|
|
|
while (true)
|
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
a = state.log.follow(a);
|
2021-11-05 02:34:35 +00:00
|
|
|
|
|
|
|
if (seenTypePacks.find(a))
|
|
|
|
break;
|
|
|
|
seenTypePacks.insert(a);
|
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (state.log.getMutable<Unifiable::Free>(a))
|
2021-11-05 02:34:35 +00:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
state.log.replace(a, Unifiable::Bound{anyTypePack});
|
2021-11-05 02:34:35 +00:00
|
|
|
}
|
2022-03-11 16:55:02 +00:00
|
|
|
else if (auto tp = state.log.getMutable<TypePack>(a))
|
2021-11-05 02:34:35 +00:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
queue.insert(queue.end(), tp->head.begin(), tp->head.end());
|
|
|
|
if (tp->tail)
|
|
|
|
a = *tp->tail;
|
|
|
|
else
|
|
|
|
break;
|
2021-11-05 02:34:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
void Unifier::tryUnifyVariadics(TypePackId subTp, TypePackId superTp, bool reversed, int subOffset)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
const VariadicTypePack* superVariadic = log.getMutable<VariadicTypePack>(superTp);
|
2022-01-07 01:46:53 +00:00
|
|
|
|
|
|
|
if (!superVariadic)
|
2021-10-29 21:25:12 +01:00
|
|
|
ice("passed non-variadic pack to tryUnifyVariadics");
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
if (const VariadicTypePack* subVariadic = get<VariadicTypePack>(subTp))
|
|
|
|
tryUnify_(reversed ? superVariadic->ty : subVariadic->ty, reversed ? subVariadic->ty : superVariadic->ty);
|
2021-10-29 21:25:12 +01:00
|
|
|
else if (get<TypePack>(subTp))
|
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
TypePackIterator subIter = begin(subTp, &log);
|
|
|
|
TypePackIterator subEnd = end(subTp);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
std::advance(subIter, subOffset);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
while (subIter != subEnd)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
tryUnify_(reversed ? superVariadic->ty : *subIter, reversed ? *subIter : superVariadic->ty);
|
|
|
|
++subIter;
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
if (std::optional<TypePackId> maybeTail = subIter.tail())
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
|
|
|
TypePackId tail = follow(*maybeTail);
|
|
|
|
if (get<FreeTypePack>(tail))
|
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
log.replace(tail, BoundTypePack(superTp));
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else if (const VariadicTypePack* vtp = get<VariadicTypePack>(tail))
|
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
tryUnify_(vtp->ty, superVariadic->ty);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else if (get<Unifiable::Generic>(tail))
|
|
|
|
{
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, GenericError{"Cannot unify variadic and generic packs"}});
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
else if (get<Unifiable::Error>(tail))
|
|
|
|
{
|
|
|
|
// Nothing to do here.
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ice("Unknown TypePack kind");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, GenericError{"Failed to unify variadic packs"}});
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-05 02:34:35 +00:00
|
|
|
static void tryUnifyWithAny(std::vector<TypeId>& queue, Unifier& state, DenseHashSet<TypeId>& seen, DenseHashSet<TypePackId>& seenTypePacks,
|
2022-02-11 19:02:09 +00:00
|
|
|
const TypeArena* typeArena, TypeId anyType, TypePackId anyTypePack)
|
2021-11-05 02:34:35 +00:00
|
|
|
{
|
|
|
|
while (!queue.empty())
|
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
TypeId ty = state.log.follow(queue.back());
|
|
|
|
queue.pop_back();
|
2022-02-11 19:02:09 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
// Types from other modules don't have free types
|
2022-03-24 22:04:14 +00:00
|
|
|
if (ty->owningArena != typeArena)
|
2022-03-11 16:55:02 +00:00
|
|
|
continue;
|
2022-02-11 19:02:09 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (seen.find(ty))
|
|
|
|
continue;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
seen.insert(ty);
|
2022-01-07 01:46:53 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (state.log.getMutable<FreeTypeVar>(ty))
|
|
|
|
{
|
|
|
|
state.log.replace(ty, BoundTypeVar{anyType});
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-03-11 16:55:02 +00:00
|
|
|
else if (auto fun = state.log.getMutable<FunctionTypeVar>(ty))
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
queueTypePack(queue, seenTypePacks, state, fun->argTypes, anyTypePack);
|
|
|
|
queueTypePack(queue, seenTypePacks, state, fun->retType, anyTypePack);
|
|
|
|
}
|
|
|
|
else if (auto table = state.log.getMutable<TableTypeVar>(ty))
|
|
|
|
{
|
|
|
|
for (const auto& [_name, prop] : table->props)
|
|
|
|
queue.push_back(prop.type);
|
2022-01-07 01:46:53 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (table->indexer)
|
2022-01-07 01:46:53 +00:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
queue.push_back(table->indexer->indexType);
|
|
|
|
queue.push_back(table->indexer->indexResultType);
|
2022-01-07 01:46:53 +00:00
|
|
|
}
|
|
|
|
}
|
2022-03-11 16:55:02 +00:00
|
|
|
else if (auto mt = state.log.getMutable<MetatableTypeVar>(ty))
|
|
|
|
{
|
|
|
|
queue.push_back(mt->table);
|
|
|
|
queue.push_back(mt->metatable);
|
|
|
|
}
|
|
|
|
else if (state.log.getMutable<ClassTypeVar>(ty))
|
|
|
|
{
|
|
|
|
// ClassTypeVars never contain free typevars.
|
|
|
|
}
|
|
|
|
else if (auto union_ = state.log.getMutable<UnionTypeVar>(ty))
|
|
|
|
queue.insert(queue.end(), union_->options.begin(), union_->options.end());
|
|
|
|
else if (auto intersection = state.log.getMutable<IntersectionTypeVar>(ty))
|
|
|
|
queue.insert(queue.end(), intersection->parts.begin(), intersection->parts.end());
|
|
|
|
else
|
|
|
|
{
|
|
|
|
} // Primitives, any, errors, and generics are left untouched.
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
void Unifier::tryUnifyWithAny(TypeId subTy, TypeId anyTy)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
LUAU_ASSERT(get<AnyTypeVar>(anyTy) || get<ErrorTypeVar>(anyTy));
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2021-12-03 06:41:04 +00:00
|
|
|
// These types are not visited in general loop below
|
2022-01-07 01:46:53 +00:00
|
|
|
if (get<PrimitiveTypeVar>(subTy) || get<AnyTypeVar>(subTy) || get<ClassTypeVar>(subTy))
|
2021-12-03 06:41:04 +00:00
|
|
|
return;
|
2021-11-05 02:34:35 +00:00
|
|
|
|
2021-12-10 22:05:05 +00:00
|
|
|
const TypePackId anyTypePack = types->addTypePack(TypePackVar{VariadicTypePack{getSingletonTypes().anyType}});
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
const TypePackId anyTP = get<AnyTypeVar>(anyTy) ? anyTypePack : types->addTypePack(TypePackVar{Unifiable::Error{}});
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
std::vector<TypeId> queue = {subTy};
|
2021-11-05 02:34:35 +00:00
|
|
|
|
2021-12-03 06:41:04 +00:00
|
|
|
sharedState.tempSeenTy.clear();
|
|
|
|
sharedState.tempSeenTp.clear();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-02-11 19:02:09 +00:00
|
|
|
Luau::tryUnifyWithAny(queue, *this, sharedState.tempSeenTy, sharedState.tempSeenTp, types, getSingletonTypes().anyType, anyTP);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
void Unifier::tryUnifyWithAny(TypePackId subTy, TypePackId anyTp)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-01-07 01:46:53 +00:00
|
|
|
LUAU_ASSERT(get<Unifiable::Error>(anyTp));
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2021-12-10 22:05:05 +00:00
|
|
|
const TypeId anyTy = getSingletonTypes().errorRecoveryType();
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2021-12-03 06:41:04 +00:00
|
|
|
std::vector<TypeId> queue;
|
2021-11-05 15:47:21 +00:00
|
|
|
|
2021-12-03 06:41:04 +00:00
|
|
|
sharedState.tempSeenTy.clear();
|
|
|
|
sharedState.tempSeenTp.clear();
|
2021-11-05 02:34:35 +00:00
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
queueTypePack(queue, sharedState.tempSeenTp, *this, subTy, anyTp);
|
2021-11-05 02:34:35 +00:00
|
|
|
|
2022-02-11 19:02:09 +00:00
|
|
|
Luau::tryUnifyWithAny(queue, *this, sharedState.tempSeenTy, sharedState.tempSeenTp, types, anyTy, anyTp);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<TypeId> Unifier::findTablePropertyRespectingMeta(TypeId lhsType, Name name)
|
|
|
|
{
|
2022-02-24 23:53:37 +00:00
|
|
|
return Luau::findTablePropertyRespectingMeta(errors, lhsType, name, location);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2022-04-15 00:57:43 +01:00
|
|
|
void Unifier::tryUnifyWithConstrainedSubTypeVar(TypeId subTy, TypeId superTy)
|
|
|
|
{
|
|
|
|
const ConstrainedTypeVar* subConstrained = get<ConstrainedTypeVar>(subTy);
|
|
|
|
if (!subConstrained)
|
|
|
|
ice("tryUnifyWithConstrainedSubTypeVar received non-ConstrainedTypeVar subTy!");
|
|
|
|
|
|
|
|
const std::vector<TypeId>& subTyParts = subConstrained->parts;
|
|
|
|
|
|
|
|
// A | B <: T if A <: T and B <: T
|
|
|
|
bool failed = false;
|
|
|
|
std::optional<TypeError> unificationTooComplex;
|
|
|
|
|
|
|
|
const size_t count = subTyParts.size();
|
|
|
|
|
|
|
|
for (size_t i = 0; i < count; ++i)
|
|
|
|
{
|
|
|
|
TypeId type = subTyParts[i];
|
|
|
|
Unifier innerState = makeChildUnifier();
|
|
|
|
innerState.tryUnify_(type, superTy);
|
|
|
|
|
|
|
|
if (i == count - 1)
|
|
|
|
log.concat(std::move(innerState.log));
|
|
|
|
|
|
|
|
++i;
|
|
|
|
|
|
|
|
if (auto e = hasUnificationTooComplex(innerState.errors))
|
|
|
|
unificationTooComplex = e;
|
|
|
|
|
|
|
|
if (!innerState.errors.empty())
|
|
|
|
{
|
|
|
|
failed = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unificationTooComplex)
|
|
|
|
reportError(*unificationTooComplex);
|
|
|
|
else if (failed)
|
|
|
|
reportError(TypeError{location, TypeMismatch{superTy, subTy}});
|
|
|
|
else
|
|
|
|
log.replace(subTy, BoundTypeVar{superTy});
|
|
|
|
}
|
|
|
|
|
|
|
|
void Unifier::tryUnifyWithConstrainedSuperTypeVar(TypeId subTy, TypeId superTy)
|
|
|
|
{
|
|
|
|
ConstrainedTypeVar* superC = log.getMutable<ConstrainedTypeVar>(superTy);
|
|
|
|
if (!superC)
|
|
|
|
ice("tryUnifyWithConstrainedSuperTypeVar received non-ConstrainedTypeVar superTy!");
|
|
|
|
|
|
|
|
// subTy could be a
|
|
|
|
// table
|
|
|
|
// metatable
|
|
|
|
// class
|
|
|
|
// function
|
|
|
|
// primitive
|
|
|
|
// free
|
|
|
|
// generic
|
|
|
|
// intersection
|
|
|
|
// union
|
|
|
|
// Do we really just tack it on? I think we might!
|
|
|
|
// We can certainly do some deduplication.
|
|
|
|
// Is there any point to deducing Player|Instance when we could just reduce to Instance?
|
|
|
|
// Is it actually ok to have multiple free types in a single intersection? What if they are later unified into the same type?
|
|
|
|
// Maybe we do a simplification step during quantification.
|
|
|
|
|
|
|
|
auto it = std::find(superC->parts.begin(), superC->parts.end(), subTy);
|
|
|
|
if (it != superC->parts.end())
|
|
|
|
return;
|
|
|
|
|
|
|
|
superC->parts.push_back(subTy);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Unifier::unifyLowerBound(TypePackId subTy, TypePackId superTy)
|
|
|
|
{
|
|
|
|
// The duplication between this and regular typepack unification is tragic.
|
|
|
|
|
|
|
|
auto superIter = begin(superTy, &log);
|
|
|
|
auto superEndIter = end(superTy);
|
|
|
|
|
|
|
|
auto subIter = begin(subTy, &log);
|
|
|
|
auto subEndIter = end(subTy);
|
|
|
|
|
|
|
|
int count = FInt::LuauTypeInferLowerBoundsIterationLimit;
|
|
|
|
|
|
|
|
for (; subIter != subEndIter; ++subIter)
|
|
|
|
{
|
|
|
|
if (0 >= --count)
|
|
|
|
ice("Internal recursion counter limit exceeded in Unifier::unifyLowerBound");
|
|
|
|
|
|
|
|
if (superIter != superEndIter)
|
|
|
|
{
|
|
|
|
tryUnify_(*subIter, *superIter);
|
|
|
|
++superIter;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (auto t = superIter.tail())
|
|
|
|
{
|
|
|
|
TypePackId tailPack = follow(*t);
|
|
|
|
|
|
|
|
if (log.get<FreeTypePack>(tailPack))
|
|
|
|
occursCheck(tailPack, subTy);
|
|
|
|
|
|
|
|
FreeTypePack* freeTailPack = log.getMutable<FreeTypePack>(tailPack);
|
|
|
|
if (!freeTailPack)
|
|
|
|
return;
|
|
|
|
|
|
|
|
TypeLevel level = freeTailPack->level;
|
|
|
|
|
|
|
|
TypePack* tp = getMutable<TypePack>(log.replace(tailPack, TypePack{}));
|
|
|
|
|
|
|
|
for (; subIter != subEndIter; ++subIter)
|
|
|
|
{
|
|
|
|
tp->head.push_back(types->addType(ConstrainedTypeVar{level, {follow(*subIter)}}));
|
|
|
|
}
|
|
|
|
|
|
|
|
tp->tail = subIter.tail();
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (superIter != superEndIter)
|
|
|
|
{
|
|
|
|
if (auto subTail = subIter.tail())
|
|
|
|
{
|
|
|
|
TypePackId subTailPack = follow(*subTail);
|
|
|
|
if (get<FreeTypePack>(subTailPack))
|
|
|
|
{
|
|
|
|
TypePack* tp = getMutable<TypePack>(log.replace(subTailPack, TypePack{}));
|
|
|
|
|
|
|
|
for (; superIter != superEndIter; ++superIter)
|
|
|
|
tp->head.push_back(*superIter);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
while (superIter != superEndIter)
|
|
|
|
{
|
|
|
|
if (!isOptional(*superIter))
|
|
|
|
{
|
|
|
|
errors.push_back(TypeError{location, CountMismatch{size(superTy), size(subTy), CountMismatch::Return}});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
++superIter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Both iters are at their respective tails
|
|
|
|
auto subTail = subIter.tail();
|
|
|
|
auto superTail = superIter.tail();
|
|
|
|
if (subTail && superTail)
|
|
|
|
tryUnify(*subTail, *superTail);
|
|
|
|
else if (subTail)
|
|
|
|
{
|
|
|
|
const FreeTypePack* freeSubTail = log.getMutable<FreeTypePack>(*subTail);
|
|
|
|
if (freeSubTail)
|
|
|
|
{
|
|
|
|
log.replace(*subTail, TypePack{});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (superTail)
|
|
|
|
{
|
|
|
|
const FreeTypePack* freeSuperTail = log.getMutable<FreeTypePack>(*superTail);
|
|
|
|
if (freeSuperTail)
|
|
|
|
{
|
|
|
|
log.replace(*superTail, TypePack{});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
void Unifier::occursCheck(TypeId needle, TypeId haystack)
|
|
|
|
{
|
2021-12-03 06:41:04 +00:00
|
|
|
sharedState.tempSeenTy.clear();
|
2021-11-05 02:34:35 +00:00
|
|
|
|
2021-12-03 06:41:04 +00:00
|
|
|
return occursCheck(sharedState.tempSeenTy, needle, haystack);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2021-12-03 06:41:04 +00:00
|
|
|
void Unifier::occursCheck(DenseHashSet<TypeId>& seen, TypeId needle, TypeId haystack)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-04-15 00:57:43 +01:00
|
|
|
RecursionLimiter _ra(&sharedState.counters.recursionCount,
|
|
|
|
FFlag::LuauAutocompleteDynamicLimits ? sharedState.counters.recursionLimit : FInt::LuauTypeInferRecursionLimit, "occursCheck for TypeId");
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-01-07 01:46:53 +00:00
|
|
|
auto check = [&](TypeId tv) {
|
|
|
|
occursCheck(seen, needle, tv);
|
|
|
|
};
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
needle = log.follow(needle);
|
|
|
|
haystack = log.follow(haystack);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (seen.find(haystack))
|
|
|
|
return;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
seen.insert(haystack);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (log.getMutable<Unifiable::Error>(needle))
|
|
|
|
return;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (!log.getMutable<Unifiable::Free>(needle))
|
|
|
|
ice("Expected needle to be free");
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (needle == haystack)
|
|
|
|
{
|
|
|
|
reportError(TypeError{location, OccursCheckFailed{}});
|
|
|
|
log.replace(needle, *getSingletonTypes().errorRecoveryType());
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
return;
|
|
|
|
}
|
2022-01-07 01:46:53 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (log.getMutable<FreeTypeVar>(haystack))
|
|
|
|
return;
|
|
|
|
else if (auto a = log.getMutable<UnionTypeVar>(haystack))
|
|
|
|
{
|
|
|
|
for (TypeId ty : a->options)
|
|
|
|
check(ty);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-03-11 16:55:02 +00:00
|
|
|
else if (auto a = log.getMutable<IntersectionTypeVar>(haystack))
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
for (TypeId ty : a->parts)
|
|
|
|
check(ty);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-04-15 00:57:43 +01:00
|
|
|
else if (auto a = log.getMutable<ConstrainedTypeVar>(haystack))
|
|
|
|
{
|
|
|
|
for (TypeId ty : a->parts)
|
|
|
|
check(ty);
|
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Unifier::occursCheck(TypePackId needle, TypePackId haystack)
|
|
|
|
{
|
2021-12-03 06:41:04 +00:00
|
|
|
sharedState.tempSeenTp.clear();
|
2021-11-05 15:47:21 +00:00
|
|
|
|
2021-12-03 06:41:04 +00:00
|
|
|
return occursCheck(sharedState.tempSeenTp, needle, haystack);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2021-12-03 06:41:04 +00:00
|
|
|
void Unifier::occursCheck(DenseHashSet<TypePackId>& seen, TypePackId needle, TypePackId haystack)
|
2021-10-29 21:25:12 +01:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
needle = log.follow(needle);
|
|
|
|
haystack = log.follow(haystack);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (seen.find(haystack))
|
|
|
|
return;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
seen.insert(haystack);
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (log.getMutable<Unifiable::Error>(needle))
|
|
|
|
return;
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (!log.getMutable<Unifiable::Free>(needle))
|
|
|
|
ice("Expected needle pack to be free");
|
2022-01-07 01:46:53 +00:00
|
|
|
|
2022-04-15 00:57:43 +01:00
|
|
|
RecursionLimiter _ra(&sharedState.counters.recursionCount,
|
|
|
|
FFlag::LuauAutocompleteDynamicLimits ? sharedState.counters.recursionLimit : FInt::LuauTypeInferRecursionLimit, "occursCheck for TypePackId");
|
2022-01-27 23:46:05 +00:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
while (!log.getMutable<ErrorTypeVar>(haystack))
|
2022-01-07 01:46:53 +00:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
if (needle == haystack)
|
|
|
|
{
|
|
|
|
reportError(TypeError{location, OccursCheckFailed{}});
|
|
|
|
log.replace(needle, *getSingletonTypes().errorRecoveryTypePack());
|
2022-01-07 01:46:53 +00:00
|
|
|
|
|
|
|
return;
|
2022-03-11 16:55:02 +00:00
|
|
|
}
|
2021-10-29 21:25:12 +01:00
|
|
|
|
2022-03-11 16:55:02 +00:00
|
|
|
if (auto a = get<TypePack>(haystack); a && a->tail)
|
2022-01-07 01:46:53 +00:00
|
|
|
{
|
2022-03-11 16:55:02 +00:00
|
|
|
haystack = log.follow(*a->tail);
|
|
|
|
continue;
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
2022-03-11 16:55:02 +00:00
|
|
|
|
|
|
|
break;
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Unifier Unifier::makeChildUnifier()
|
|
|
|
{
|
2022-05-20 01:02:24 +01:00
|
|
|
Unifier u = Unifier{types, mode, location, variance, sharedState, &log};
|
2022-04-15 00:57:43 +01:00
|
|
|
u.anyIsTop = anyIsTop;
|
|
|
|
return u;
|
|
|
|
}
|
|
|
|
|
|
|
|
// A utility function that appends the given error to the unifier's error log.
|
|
|
|
// This allows setting a breakpoint wherever the unifier reports an error.
|
|
|
|
void Unifier::reportError(TypeError err)
|
|
|
|
{
|
|
|
|
errors.push_back(std::move(err));
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Unifier::isNonstrictMode() const
|
|
|
|
{
|
|
|
|
return (mode == Mode::Nonstrict) || (mode == Mode::NoCheck);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Unifier::checkChildUnifierTypeMismatch(const ErrorVec& innerErrors, TypeId wantedType, TypeId givenType)
|
|
|
|
{
|
|
|
|
if (auto e = hasUnificationTooComplex(innerErrors))
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(*e);
|
2021-10-29 21:25:12 +01:00
|
|
|
else if (!innerErrors.empty())
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(TypeError{location, TypeMismatch{wantedType, givenType}});
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
2021-11-12 14:27:34 +00:00
|
|
|
void Unifier::checkChildUnifierTypeMismatch(const ErrorVec& innerErrors, const std::string& prop, TypeId wantedType, TypeId givenType)
|
|
|
|
{
|
|
|
|
if (auto e = hasUnificationTooComplex(innerErrors))
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(*e);
|
2021-11-12 14:27:34 +00:00
|
|
|
else if (!innerErrors.empty())
|
2022-01-27 23:46:05 +00:00
|
|
|
reportError(
|
2021-12-10 22:05:05 +00:00
|
|
|
TypeError{location, TypeMismatch{wantedType, givenType, format("Property '%s' is not compatible.", prop.c_str()), innerErrors.front()}});
|
2021-11-12 14:27:34 +00:00
|
|
|
}
|
|
|
|
|
2021-10-29 21:25:12 +01:00
|
|
|
void Unifier::ice(const std::string& message, const Location& location)
|
|
|
|
{
|
2021-11-05 15:47:21 +00:00
|
|
|
sharedState.iceHandler->ice(message, location);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Unifier::ice(const std::string& message)
|
|
|
|
{
|
2021-11-05 15:47:21 +00:00
|
|
|
sharedState.iceHandler->ice(message);
|
2021-10-29 21:25:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace Luau
|