2024-02-02 18:20:03 +00:00
|
|
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
|
|
#include "Luau/OverloadResolution.h"
|
|
|
|
|
2024-03-08 23:57:12 +00:00
|
|
|
#include "Luau/Instantiation2.h"
|
2024-02-02 18:20:03 +00:00
|
|
|
#include "Luau/Subtyping.h"
|
|
|
|
#include "Luau/TxnLog.h"
|
|
|
|
#include "Luau/Type.h"
|
2024-07-11 23:13:45 +01:00
|
|
|
#include "Luau/TypeFunction.h"
|
2024-02-02 18:20:03 +00:00
|
|
|
#include "Luau/TypePack.h"
|
|
|
|
#include "Luau/TypeUtils.h"
|
2024-03-08 23:57:12 +00:00
|
|
|
#include "Luau/Unifier2.h"
|
2024-02-02 18:20:03 +00:00
|
|
|
|
|
|
|
namespace Luau
|
|
|
|
{
|
|
|
|
|
|
|
|
OverloadResolver::OverloadResolver(NotNull<BuiltinTypes> builtinTypes, NotNull<TypeArena> arena, NotNull<Normalizer> normalizer, NotNull<Scope> scope,
|
|
|
|
NotNull<InternalErrorReporter> reporter, NotNull<TypeCheckLimits> limits, Location callLocation)
|
|
|
|
: builtinTypes(builtinTypes)
|
|
|
|
, arena(arena)
|
|
|
|
, normalizer(normalizer)
|
|
|
|
, scope(scope)
|
|
|
|
, ice(reporter)
|
|
|
|
, limits(limits)
|
|
|
|
, subtyping({builtinTypes, arena, normalizer, ice, scope})
|
|
|
|
, callLoc(callLocation)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
std::pair<OverloadResolver::Analysis, TypeId> OverloadResolver::selectOverload(TypeId ty, TypePackId argsPack)
|
|
|
|
{
|
2024-03-08 23:57:12 +00:00
|
|
|
auto tryOne = [&](TypeId f) {
|
|
|
|
if (auto ftv = get<FunctionType>(f))
|
|
|
|
{
|
|
|
|
SubtypingResult r = subtyping.isSubtype(argsPack, ftv->argTypes);
|
|
|
|
if (r.isSubtype)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
};
|
|
|
|
|
2024-02-02 18:20:03 +00:00
|
|
|
TypeId t = follow(ty);
|
2024-03-08 23:57:12 +00:00
|
|
|
|
|
|
|
if (tryOne(ty))
|
|
|
|
return {Analysis::Ok, ty};
|
|
|
|
|
2024-02-02 18:20:03 +00:00
|
|
|
if (auto it = get<IntersectionType>(t))
|
|
|
|
{
|
|
|
|
for (TypeId component : it)
|
|
|
|
{
|
2024-03-08 23:57:12 +00:00
|
|
|
if (tryOne(component))
|
|
|
|
return {Analysis::Ok, component};
|
2024-02-02 18:20:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return {Analysis::OverloadIsNonviable, ty};
|
|
|
|
}
|
|
|
|
|
|
|
|
void OverloadResolver::resolve(TypeId fnTy, const TypePack* args, AstExpr* selfExpr, const std::vector<AstExpr*>* argExprs)
|
|
|
|
{
|
|
|
|
fnTy = follow(fnTy);
|
|
|
|
|
|
|
|
auto it = get<IntersectionType>(fnTy);
|
|
|
|
if (!it)
|
|
|
|
{
|
|
|
|
auto [analysis, errors] = checkOverload(fnTy, args, selfExpr, argExprs);
|
|
|
|
add(analysis, fnTy, std::move(errors));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (TypeId ty : it)
|
|
|
|
{
|
|
|
|
if (resolution.find(ty) != resolution.end())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
auto [analysis, errors] = checkOverload(ty, args, selfExpr, argExprs);
|
|
|
|
add(analysis, ty, std::move(errors));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<ErrorVec> OverloadResolver::testIsSubtype(const Location& location, TypeId subTy, TypeId superTy)
|
|
|
|
{
|
|
|
|
auto r = subtyping.isSubtype(subTy, superTy);
|
|
|
|
ErrorVec errors;
|
|
|
|
|
|
|
|
if (r.normalizationTooComplex)
|
|
|
|
errors.emplace_back(location, NormalizationTooComplex{});
|
|
|
|
|
|
|
|
if (!r.isSubtype)
|
|
|
|
{
|
|
|
|
switch (shouldSuppressErrors(normalizer, subTy).orElse(shouldSuppressErrors(normalizer, superTy)))
|
|
|
|
{
|
|
|
|
case ErrorSuppression::Suppress:
|
|
|
|
break;
|
|
|
|
case ErrorSuppression::NormalizationFailed:
|
|
|
|
errors.emplace_back(location, NormalizationTooComplex{});
|
|
|
|
// intentionally fallthrough here since we couldn't prove this was error-suppressing
|
|
|
|
case ErrorSuppression::DoNotSuppress:
|
|
|
|
errors.emplace_back(location, TypeMismatch{superTy, subTy});
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (errors.empty())
|
|
|
|
return std::nullopt;
|
|
|
|
|
|
|
|
return errors;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<ErrorVec> OverloadResolver::testIsSubtype(const Location& location, TypePackId subTy, TypePackId superTy)
|
|
|
|
{
|
|
|
|
auto r = subtyping.isSubtype(subTy, superTy);
|
|
|
|
ErrorVec errors;
|
|
|
|
|
|
|
|
if (r.normalizationTooComplex)
|
|
|
|
errors.emplace_back(location, NormalizationTooComplex{});
|
|
|
|
|
|
|
|
if (!r.isSubtype)
|
|
|
|
{
|
|
|
|
switch (shouldSuppressErrors(normalizer, subTy).orElse(shouldSuppressErrors(normalizer, superTy)))
|
|
|
|
{
|
|
|
|
case ErrorSuppression::Suppress:
|
|
|
|
break;
|
|
|
|
case ErrorSuppression::NormalizationFailed:
|
|
|
|
errors.emplace_back(location, NormalizationTooComplex{});
|
|
|
|
// intentionally fallthrough here since we couldn't prove this was error-suppressing
|
|
|
|
case ErrorSuppression::DoNotSuppress:
|
|
|
|
errors.emplace_back(location, TypePackMismatch{superTy, subTy});
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (errors.empty())
|
|
|
|
return std::nullopt;
|
|
|
|
|
|
|
|
return errors;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::pair<OverloadResolver::Analysis, ErrorVec> OverloadResolver::checkOverload(
|
|
|
|
TypeId fnTy, const TypePack* args, AstExpr* fnLoc, const std::vector<AstExpr*>* argExprs, bool callMetamethodOk)
|
|
|
|
{
|
|
|
|
fnTy = follow(fnTy);
|
|
|
|
|
|
|
|
ErrorVec discard;
|
|
|
|
if (get<AnyType>(fnTy) || get<ErrorType>(fnTy) || get<NeverType>(fnTy))
|
|
|
|
return {Ok, {}};
|
|
|
|
else if (auto fn = get<FunctionType>(fnTy))
|
|
|
|
return checkOverload_(fnTy, fn, args, fnLoc, argExprs); // Intentionally split to reduce the stack pressure of this function.
|
|
|
|
else if (auto callMm = findMetatableEntry(builtinTypes, discard, fnTy, "__call", callLoc); callMm && callMetamethodOk)
|
|
|
|
{
|
|
|
|
// Calling a metamethod forwards the `fnTy` as self.
|
|
|
|
TypePack withSelf = *args;
|
|
|
|
withSelf.head.insert(withSelf.head.begin(), fnTy);
|
|
|
|
|
|
|
|
std::vector<AstExpr*> withSelfExprs = *argExprs;
|
|
|
|
withSelfExprs.insert(withSelfExprs.begin(), fnLoc);
|
|
|
|
|
|
|
|
return checkOverload(*callMm, &withSelf, fnLoc, &withSelfExprs, /*callMetamethodOk=*/false);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return {TypeIsNotAFunction, {}}; // Intentionally empty. We can just fabricate the type error later on.
|
|
|
|
}
|
|
|
|
|
|
|
|
bool OverloadResolver::isLiteral(AstExpr* expr)
|
|
|
|
{
|
|
|
|
if (auto group = expr->as<AstExprGroup>())
|
|
|
|
return isLiteral(group->expr);
|
|
|
|
else if (auto assertion = expr->as<AstExprTypeAssertion>())
|
|
|
|
return isLiteral(assertion->expr);
|
|
|
|
|
|
|
|
return expr->is<AstExprConstantNil>() || expr->is<AstExprConstantBool>() || expr->is<AstExprConstantNumber>() ||
|
|
|
|
expr->is<AstExprConstantString>() || expr->is<AstExprFunction>() || expr->is<AstExprTable>();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::pair<OverloadResolver::Analysis, ErrorVec> OverloadResolver::checkOverload_(
|
|
|
|
TypeId fnTy, const FunctionType* fn, const TypePack* args, AstExpr* fnExpr, const std::vector<AstExpr*>* argExprs)
|
|
|
|
{
|
2024-07-11 23:13:45 +01:00
|
|
|
FunctionGraphReductionResult result =
|
|
|
|
reduceTypeFunctions(fnTy, callLoc, TypeFunctionContext{arena, builtinTypes, scope, normalizer, ice, limits}, /*force=*/true);
|
2024-02-02 18:20:03 +00:00
|
|
|
if (!result.errors.empty())
|
|
|
|
return {OverloadIsNonviable, result.errors};
|
|
|
|
|
|
|
|
ErrorVec argumentErrors;
|
2024-07-11 23:13:45 +01:00
|
|
|
TypePackId typ = arena->addTypePack(*args);
|
2024-02-02 18:20:03 +00:00
|
|
|
|
2024-07-11 23:13:45 +01:00
|
|
|
TypeId prospectiveFunction = arena->addType(FunctionType{typ, builtinTypes->anyTypePack});
|
2024-02-02 18:20:03 +00:00
|
|
|
SubtypingResult sr = subtyping.isSubtype(fnTy, prospectiveFunction);
|
|
|
|
|
|
|
|
if (sr.isSubtype)
|
|
|
|
return {Analysis::Ok, {}};
|
|
|
|
|
|
|
|
if (1 == sr.reasoning.size())
|
|
|
|
{
|
|
|
|
const SubtypingReasoning& reason = *sr.reasoning.begin();
|
|
|
|
|
|
|
|
const TypePath::Path justArguments{TypePath::PackField::Arguments};
|
|
|
|
|
|
|
|
if (reason.subPath == justArguments && reason.superPath == justArguments)
|
|
|
|
{
|
|
|
|
// If the subtype test failed only due to an arity mismatch,
|
|
|
|
// it is still possible that this function call is okay.
|
|
|
|
// Subtype testing does not know anything about optional
|
|
|
|
// function arguments.
|
|
|
|
//
|
|
|
|
// This can only happen if the actual function call has a
|
|
|
|
// finite set of arguments which is too short for the
|
|
|
|
// function being called. If all of those unsatisfied
|
|
|
|
// function arguments are options, then this function call
|
|
|
|
// is ok.
|
|
|
|
|
|
|
|
const size_t firstUnsatisfiedArgument = argExprs->size();
|
|
|
|
const auto [requiredHead, _requiredTail] = flatten(fn->argTypes);
|
|
|
|
|
|
|
|
// If too many arguments were supplied, this overload
|
|
|
|
// definitely does not match.
|
|
|
|
if (args->head.size() > requiredHead.size())
|
|
|
|
{
|
|
|
|
auto [minParams, optMaxParams] = getParameterExtents(TxnLog::empty(), fn->argTypes);
|
|
|
|
TypeError error{fnExpr->location, CountMismatch{minParams, optMaxParams, args->head.size(), CountMismatch::Arg, false}};
|
|
|
|
|
|
|
|
return {Analysis::ArityMismatch, {error}};
|
|
|
|
}
|
|
|
|
|
|
|
|
// If any of the unsatisfied arguments are not supertypes of
|
|
|
|
// nil, then this overload does not match.
|
|
|
|
for (size_t i = firstUnsatisfiedArgument; i < requiredHead.size(); ++i)
|
|
|
|
{
|
|
|
|
if (!subtyping.isSubtype(builtinTypes->nilType, requiredHead[i]).isSubtype)
|
|
|
|
{
|
|
|
|
auto [minParams, optMaxParams] = getParameterExtents(TxnLog::empty(), fn->argTypes);
|
|
|
|
TypeError error{fnExpr->location, CountMismatch{minParams, optMaxParams, args->head.size(), CountMismatch::Arg, false}};
|
|
|
|
|
|
|
|
return {Analysis::ArityMismatch, {error}};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return {Analysis::Ok, {}};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ErrorVec errors;
|
|
|
|
|
|
|
|
for (const SubtypingReasoning& reason : sr.reasoning)
|
|
|
|
{
|
|
|
|
/* The return type of our prospective function is always
|
|
|
|
* any... so any subtype failures here can only arise from
|
|
|
|
* argument type mismatches.
|
|
|
|
*/
|
|
|
|
|
|
|
|
Location argLocation;
|
2024-03-01 13:58:44 +00:00
|
|
|
if (reason.superPath.components.size() <= 1)
|
|
|
|
break;
|
2024-02-02 18:20:03 +00:00
|
|
|
|
|
|
|
if (const Luau::TypePath::Index* pathIndexComponent = get_if<Luau::TypePath::Index>(&reason.superPath.components.at(1)))
|
|
|
|
{
|
|
|
|
size_t nthArgument = pathIndexComponent->index;
|
2024-07-11 23:13:45 +01:00
|
|
|
// if the nth type argument to the function is less than the number of ast expressions we passed to the function
|
|
|
|
// we should be able to pull out the location of the argument
|
|
|
|
// If the nth type argument to the function is out of range of the ast expressions we passed to the function
|
|
|
|
// e.g. table.pack(functionThatReturnsMultipleArguments(arg1, arg2, ....)), default to the location of the last passed expression
|
|
|
|
// If we passed no expression arguments to the call, default to the location of the function expression.
|
|
|
|
argLocation = nthArgument < argExprs->size() ? argExprs->at(nthArgument)->location
|
|
|
|
: argExprs->size() != 0 ? argExprs->back()->location
|
|
|
|
: fnExpr->location;
|
2024-02-02 18:20:03 +00:00
|
|
|
|
|
|
|
std::optional<TypeId> failedSubTy = traverseForType(fnTy, reason.subPath, builtinTypes);
|
|
|
|
std::optional<TypeId> failedSuperTy = traverseForType(prospectiveFunction, reason.superPath, builtinTypes);
|
|
|
|
|
|
|
|
if (failedSubTy && failedSuperTy)
|
|
|
|
{
|
|
|
|
|
|
|
|
switch (shouldSuppressErrors(normalizer, *failedSubTy).orElse(shouldSuppressErrors(normalizer, *failedSuperTy)))
|
|
|
|
{
|
|
|
|
case ErrorSuppression::Suppress:
|
|
|
|
break;
|
|
|
|
case ErrorSuppression::NormalizationFailed:
|
|
|
|
errors.emplace_back(argLocation, NormalizationTooComplex{});
|
|
|
|
// intentionally fallthrough here since we couldn't prove this was error-suppressing
|
|
|
|
case ErrorSuppression::DoNotSuppress:
|
|
|
|
// TODO extract location from the SubtypingResult path and argExprs
|
|
|
|
switch (reason.variance)
|
|
|
|
{
|
|
|
|
case SubtypingVariance::Covariant:
|
|
|
|
case SubtypingVariance::Contravariant:
|
|
|
|
errors.emplace_back(argLocation, TypeMismatch{*failedSubTy, *failedSuperTy, TypeMismatch::CovariantContext});
|
|
|
|
break;
|
|
|
|
case SubtypingVariance::Invariant:
|
|
|
|
errors.emplace_back(argLocation, TypeMismatch{*failedSubTy, *failedSuperTy, TypeMismatch::InvariantContext});
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
LUAU_ASSERT(0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<TypePackId> failedSubPack = traverseForPack(fnTy, reason.subPath, builtinTypes);
|
|
|
|
std::optional<TypePackId> failedSuperPack = traverseForPack(prospectiveFunction, reason.superPath, builtinTypes);
|
|
|
|
|
|
|
|
if (failedSubPack && failedSuperPack)
|
|
|
|
{
|
2024-03-15 21:01:00 +00:00
|
|
|
// If a bug in type inference occurs, we may have a mismatch in the return packs.
|
|
|
|
// This happens when inference incorrectly leaves the result type of a function free.
|
|
|
|
// If this happens, we don't want to explode, so we'll use the function's location.
|
|
|
|
if (argExprs->empty())
|
|
|
|
argLocation = fnExpr->location;
|
|
|
|
else
|
|
|
|
argLocation = argExprs->at(argExprs->size() - 1)->location;
|
2024-02-02 18:20:03 +00:00
|
|
|
|
|
|
|
// TODO extract location from the SubtypingResult path and argExprs
|
|
|
|
switch (reason.variance)
|
|
|
|
{
|
|
|
|
case SubtypingVariance::Covariant:
|
|
|
|
errors.emplace_back(argLocation, TypePackMismatch{*failedSubPack, *failedSuperPack});
|
|
|
|
break;
|
|
|
|
case SubtypingVariance::Contravariant:
|
|
|
|
errors.emplace_back(argLocation, TypePackMismatch{*failedSuperPack, *failedSubPack});
|
|
|
|
break;
|
|
|
|
case SubtypingVariance::Invariant:
|
|
|
|
errors.emplace_back(argLocation, TypePackMismatch{*failedSubPack, *failedSuperPack});
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
LUAU_ASSERT(0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return {Analysis::OverloadIsNonviable, std::move(errors)};
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t OverloadResolver::indexof(Analysis analysis)
|
|
|
|
{
|
|
|
|
switch (analysis)
|
|
|
|
{
|
|
|
|
case Ok:
|
|
|
|
return ok.size();
|
|
|
|
case TypeIsNotAFunction:
|
|
|
|
return nonFunctions.size();
|
|
|
|
case ArityMismatch:
|
|
|
|
return arityMismatches.size();
|
|
|
|
case OverloadIsNonviable:
|
|
|
|
return nonviableOverloads.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
ice->ice("Inexhaustive switch in FunctionCallResolver::indexof");
|
|
|
|
}
|
|
|
|
|
|
|
|
void OverloadResolver::add(Analysis analysis, TypeId ty, ErrorVec&& errors)
|
|
|
|
{
|
|
|
|
resolution.insert(ty, {analysis, indexof(analysis)});
|
|
|
|
|
|
|
|
switch (analysis)
|
|
|
|
{
|
|
|
|
case Ok:
|
|
|
|
LUAU_ASSERT(errors.empty());
|
|
|
|
ok.push_back(ty);
|
|
|
|
break;
|
|
|
|
case TypeIsNotAFunction:
|
|
|
|
LUAU_ASSERT(errors.empty());
|
|
|
|
nonFunctions.push_back(ty);
|
|
|
|
break;
|
|
|
|
case ArityMismatch:
|
|
|
|
LUAU_ASSERT(!errors.empty());
|
|
|
|
arityMismatches.emplace_back(ty, std::move(errors));
|
|
|
|
break;
|
|
|
|
case OverloadIsNonviable:
|
|
|
|
nonviableOverloads.emplace_back(ty, std::move(errors));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-30 22:49:03 +00:00
|
|
|
// we wrap calling the overload resolver in a separate function to reduce overall stack pressure in `solveFunctionCall`.
|
|
|
|
// this limits the lifetime of `OverloadResolver`, a large type, to only as long as it is actually needed.
|
2024-04-19 22:04:30 +01:00
|
|
|
std::optional<TypeId> selectOverload(NotNull<BuiltinTypes> builtinTypes, NotNull<TypeArena> arena, NotNull<Normalizer> normalizer,
|
|
|
|
NotNull<Scope> scope, NotNull<InternalErrorReporter> iceReporter, NotNull<TypeCheckLimits> limits, const Location& location, TypeId fn,
|
|
|
|
TypePackId argsPack)
|
2024-03-30 22:49:03 +00:00
|
|
|
{
|
2024-04-19 22:04:30 +01:00
|
|
|
OverloadResolver resolver{builtinTypes, arena, normalizer, scope, iceReporter, limits, location};
|
|
|
|
auto [status, overload] = resolver.selectOverload(fn, argsPack);
|
2024-03-30 22:49:03 +00:00
|
|
|
|
2024-04-19 22:04:30 +01:00
|
|
|
if (status == OverloadResolver::Analysis::Ok)
|
|
|
|
return overload;
|
2024-03-30 22:49:03 +00:00
|
|
|
|
2024-04-19 22:04:30 +01:00
|
|
|
if (get<AnyType>(fn) || get<FreeType>(fn))
|
|
|
|
return fn;
|
2024-03-30 22:49:03 +00:00
|
|
|
|
2024-04-19 22:04:30 +01:00
|
|
|
return {};
|
2024-03-30 22:49:03 +00:00
|
|
|
}
|
2024-02-02 18:20:03 +00:00
|
|
|
|
2024-04-19 22:04:30 +01:00
|
|
|
SolveResult solveFunctionCall(NotNull<TypeArena> arena, NotNull<BuiltinTypes> builtinTypes, NotNull<Normalizer> normalizer,
|
|
|
|
NotNull<InternalErrorReporter> iceReporter, NotNull<TypeCheckLimits> limits, NotNull<Scope> scope, const Location& location, TypeId fn,
|
|
|
|
TypePackId argsPack)
|
2024-03-08 23:57:12 +00:00
|
|
|
{
|
2024-03-30 22:49:03 +00:00
|
|
|
std::optional<TypeId> overloadToUse = selectOverload(builtinTypes, arena, normalizer, scope, iceReporter, limits, location, fn, argsPack);
|
|
|
|
if (!overloadToUse)
|
2024-03-08 23:57:12 +00:00
|
|
|
return {SolveResult::NoMatchingOverload};
|
|
|
|
|
|
|
|
TypePackId resultPack = arena->freshTypePack(scope);
|
|
|
|
|
|
|
|
TypeId inferredTy = arena->addType(FunctionType{TypeLevel{}, scope.get(), argsPack, resultPack});
|
|
|
|
Unifier2 u2{NotNull{arena}, builtinTypes, scope, iceReporter};
|
|
|
|
|
2024-03-30 22:49:03 +00:00
|
|
|
const bool occursCheckPassed = u2.unify(*overloadToUse, inferredTy);
|
2024-03-08 23:57:12 +00:00
|
|
|
|
|
|
|
if (!u2.genericSubstitutions.empty() || !u2.genericPackSubstitutions.empty())
|
|
|
|
{
|
|
|
|
Instantiation2 instantiation{arena, std::move(u2.genericSubstitutions), std::move(u2.genericPackSubstitutions)};
|
|
|
|
|
|
|
|
std::optional<TypePackId> subst = instantiation.substitute(resultPack);
|
|
|
|
|
|
|
|
if (!subst)
|
|
|
|
return {SolveResult::CodeTooComplex};
|
|
|
|
else
|
|
|
|
resultPack = *subst;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!occursCheckPassed)
|
|
|
|
return {SolveResult::OccursCheckFailed};
|
|
|
|
|
|
|
|
SolveResult result;
|
|
|
|
result.result = SolveResult::Ok;
|
|
|
|
result.typePackId = resultPack;
|
|
|
|
|
|
|
|
LUAU_ASSERT(overloadToUse);
|
2024-03-30 22:49:03 +00:00
|
|
|
result.overloadToUse = *overloadToUse;
|
2024-03-08 23:57:12 +00:00
|
|
|
result.inferredTy = inferredTy;
|
|
|
|
result.expandedFreeTypes = std::move(u2.expandedFreeTypes);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2024-02-02 18:20:03 +00:00
|
|
|
} // namespace Luau
|