2022-05-27 06:08:16 +08:00
|
|
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
|
|
#pragma once
|
|
|
|
|
2023-08-05 03:18:54 +08:00
|
|
|
#include "Luau/NotNull.h"
|
2022-05-27 06:08:16 +08:00
|
|
|
#include "Luau/Substitution.h"
|
2023-10-21 09:10:30 +08:00
|
|
|
#include "Luau/TypeFwd.h"
|
2022-05-27 06:08:16 +08:00
|
|
|
#include "Luau/Unifiable.h"
|
|
|
|
|
|
|
|
namespace Luau
|
|
|
|
{
|
|
|
|
|
|
|
|
struct TxnLog;
|
2023-08-05 03:18:54 +08:00
|
|
|
struct TypeArena;
|
|
|
|
struct TypeCheckLimits;
|
2022-05-27 06:08:16 +08:00
|
|
|
|
|
|
|
// A substitution which replaces generic types in a given set by free types.
|
|
|
|
struct ReplaceGenerics : Substitution
|
|
|
|
{
|
2023-09-16 01:26:59 +08:00
|
|
|
ReplaceGenerics(const TxnLog* log, TypeArena* arena, NotNull<BuiltinTypes> builtinTypes, TypeLevel level, Scope* scope,
|
|
|
|
const std::vector<TypeId>& generics, const std::vector<TypePackId>& genericPacks)
|
2022-05-27 06:08:16 +08:00
|
|
|
: Substitution(log, arena)
|
2023-08-05 03:18:54 +08:00
|
|
|
, builtinTypes(builtinTypes)
|
2022-05-27 06:08:16 +08:00
|
|
|
, level(level)
|
2022-09-30 06:23:10 +08:00
|
|
|
, scope(scope)
|
2022-05-27 06:08:16 +08:00
|
|
|
, generics(generics)
|
|
|
|
, genericPacks(genericPacks)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2024-06-21 07:37:55 +08:00
|
|
|
void resetState(const TxnLog* log, TypeArena* arena, NotNull<BuiltinTypes> builtinTypes, TypeLevel level, Scope* scope,
|
|
|
|
const std::vector<TypeId>& generics, const std::vector<TypePackId>& genericPacks);
|
|
|
|
|
2023-08-05 03:18:54 +08:00
|
|
|
NotNull<BuiltinTypes> builtinTypes;
|
|
|
|
|
2022-05-27 06:08:16 +08:00
|
|
|
TypeLevel level;
|
2022-09-30 06:23:10 +08:00
|
|
|
Scope* scope;
|
2022-05-27 06:08:16 +08:00
|
|
|
std::vector<TypeId> generics;
|
|
|
|
std::vector<TypePackId> genericPacks;
|
2024-06-21 07:37:55 +08:00
|
|
|
|
2022-05-27 06:08:16 +08:00
|
|
|
bool ignoreChildren(TypeId ty) override;
|
|
|
|
bool isDirty(TypeId ty) override;
|
|
|
|
bool isDirty(TypePackId tp) override;
|
|
|
|
TypeId clean(TypeId ty) override;
|
|
|
|
TypePackId clean(TypePackId tp) override;
|
|
|
|
};
|
|
|
|
|
|
|
|
// A substitution which replaces generic functions by monomorphic functions
|
|
|
|
struct Instantiation : Substitution
|
|
|
|
{
|
2023-08-05 03:18:54 +08:00
|
|
|
Instantiation(const TxnLog* log, TypeArena* arena, NotNull<BuiltinTypes> builtinTypes, TypeLevel level, Scope* scope)
|
2022-05-27 06:08:16 +08:00
|
|
|
: Substitution(log, arena)
|
2023-08-05 03:18:54 +08:00
|
|
|
, builtinTypes(builtinTypes)
|
2022-05-27 06:08:16 +08:00
|
|
|
, level(level)
|
2022-09-30 06:23:10 +08:00
|
|
|
, scope(scope)
|
2024-06-21 07:37:55 +08:00
|
|
|
, reusableReplaceGenerics(log, arena, builtinTypes, level, scope, {}, {})
|
2022-05-27 06:08:16 +08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2024-06-21 07:37:55 +08:00
|
|
|
void resetState(const TxnLog* log, TypeArena* arena, NotNull<BuiltinTypes> builtinTypes, TypeLevel level, Scope* scope);
|
|
|
|
|
2023-08-05 03:18:54 +08:00
|
|
|
NotNull<BuiltinTypes> builtinTypes;
|
|
|
|
|
2022-05-27 06:08:16 +08:00
|
|
|
TypeLevel level;
|
2022-09-30 06:23:10 +08:00
|
|
|
Scope* scope;
|
2024-06-21 07:37:55 +08:00
|
|
|
|
|
|
|
ReplaceGenerics reusableReplaceGenerics;
|
|
|
|
|
2022-05-27 06:08:16 +08:00
|
|
|
bool ignoreChildren(TypeId ty) override;
|
|
|
|
bool isDirty(TypeId ty) override;
|
|
|
|
bool isDirty(TypePackId tp) override;
|
|
|
|
TypeId clean(TypeId ty) override;
|
|
|
|
TypePackId clean(TypePackId tp) override;
|
|
|
|
};
|
|
|
|
|
2023-08-05 03:18:54 +08:00
|
|
|
/** Attempt to instantiate a type. Only used under local type inference.
|
|
|
|
*
|
|
|
|
* When given a generic function type, instantiate() will return a copy with the
|
|
|
|
* generics replaced by fresh types. Instantiation will return the same TypeId
|
|
|
|
* back if the function does not have any generics.
|
|
|
|
*
|
|
|
|
* All higher order generics are left as-is. For example, instantiation of
|
|
|
|
* <X>(<Y>(Y) -> (X, Y)) -> (X, Y) is (<Y>(Y) -> ('x, Y)) -> ('x, Y)
|
|
|
|
*
|
|
|
|
* We substitute the generic X for the free 'x, but leave the generic Y alone.
|
|
|
|
*
|
|
|
|
* Instantiation fails only when processing the type causes internal recursion
|
|
|
|
* limits to be exceeded.
|
|
|
|
*/
|
2023-09-16 01:26:59 +08:00
|
|
|
std::optional<TypeId> instantiate(
|
|
|
|
NotNull<BuiltinTypes> builtinTypes, NotNull<TypeArena> arena, NotNull<TypeCheckLimits> limits, NotNull<Scope> scope, TypeId ty);
|
2023-08-05 03:18:54 +08:00
|
|
|
|
2022-05-27 06:08:16 +08:00
|
|
|
} // namespace Luau
|