mirror of
https://github.com/luau-lang/luau.git
synced 2025-04-05 11:20:54 +01:00
## New Type Solver 1. Update resolved types for singleton unions and intersections to avoid crashing when type checking type assertions. 2. Generalize free return type pack of a function type inferred at call site to ensure that the free type does not leak to another module. 3. Fix crash from cyclic indexers by reducing if possible or producing an error otherwise. 4. Fix handling of irreducible type functions to prevent type inference from failing. 5. Fix handling of recursive metatables to avoid infinite recursion. ## New and Old Type Solver Fix accidental capture of all exceptions in multi-threaded typechecking by converting all typechecking exceptions to `InternalCompilerError` and only capturing those. ## Fragment Autocomplete 1. Add a block based diff algorithm based on class index and span for re-typechecking. This reduces the granularity of fragment autocomplete to avoid flakiness when the fragment does not have enough type information. 2. Fix bugs arising from incorrect scope selection for autocompletion. ## Roundtrippable AST Store type alias location in `TypeFun` class to ensure it is accessible for exported types as part of the public interface. ## Build System 1. Bump minimum supported CMake version to 3.10 since GitHub is phasing out the currently supported minimum version 3.0, released 11 years ago. 2. Fix compilation when `HARDSTACKTESTS` is enabled. ## Miscellaneous Flag removals and cleanup of unused code. ## Internal Contributors Co-authored-by: Andy Friesen <afriesen@roblox.com> Co-authored-by: Ariel Weiss <aaronweiss@roblox.com> Co-authored-by: Hunter Goldstein <hgoldstein@roblox.com> Co-authored-by: Talha Pathan <tpathan@roblox.com> Co-authored-by: Vighnesh Vijay <vvijay@roblox.com> Co-authored-by: Vyacheslav Egorov <vegorov@roblox.com> ## External Contributors Thanks to [@grh-official](https://github.com/grh-official) for PR #1759 **Full Changelog**: https://github.com/luau-lang/luau/compare/0.667...0.668 --------- Co-authored-by: Hunter Goldstein <hgoldstein@roblox.com> Co-authored-by: Varun Saini <61795485+vrn-sn@users.noreply.github.com> Co-authored-by: Alexander Youngblood <ayoungblood@roblox.com> Co-authored-by: Menarul Alam <malam@roblox.com> Co-authored-by: Vighnesh <vvijay@roblox.com> Co-authored-by: Vyacheslav Egorov <vegorov@roblox.com> Co-authored-by: Ariel Weiss <aaronweiss@roblox.com>
311 lines
11 KiB
C++
311 lines
11 KiB
C++
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
#pragma once
|
|
|
|
#include "Luau/Type.h"
|
|
#include "Luau/TypePack.h"
|
|
|
|
#include <memory>
|
|
#include <unordered_map>
|
|
|
|
namespace Luau
|
|
{
|
|
|
|
using TypeOrPackId = const void*;
|
|
|
|
// Pending state for a Type. Generated by a TxnLog and committed via
|
|
// TxnLog::commit.
|
|
struct PendingType
|
|
{
|
|
// The pending Type state.
|
|
Type pending;
|
|
|
|
// On very rare occasions, we need to delete an entry from the TxnLog.
|
|
// DenseHashMap does not afford that so we note its deadness here.
|
|
bool dead = false;
|
|
|
|
explicit PendingType(Type state)
|
|
: pending(std::move(state))
|
|
{
|
|
}
|
|
};
|
|
|
|
std::string toString(PendingType* pending);
|
|
std::string dump(PendingType* pending);
|
|
|
|
// Pending state for a TypePackVar. Generated by a TxnLog and committed via
|
|
// TxnLog::commit.
|
|
struct PendingTypePack
|
|
{
|
|
// The pending TypePackVar state.
|
|
TypePackVar pending;
|
|
|
|
explicit PendingTypePack(TypePackVar state)
|
|
: pending(std::move(state))
|
|
{
|
|
}
|
|
};
|
|
|
|
std::string toString(PendingTypePack* pending);
|
|
std::string dump(PendingTypePack* pending);
|
|
|
|
template<typename T>
|
|
T* getMutable(PendingType* pending)
|
|
{
|
|
// We use getMutable here because this state is intended to be mutated freely.
|
|
return getMutable<T>(&pending->pending);
|
|
}
|
|
|
|
template<typename T>
|
|
T* getMutable(PendingTypePack* pending)
|
|
{
|
|
// We use getMutable here because this state is intended to be mutated freely.
|
|
return getMutable<T>(&pending->pending);
|
|
}
|
|
|
|
// Log of what TypeIds we are rebinding, to be committed later.
|
|
struct TxnLog
|
|
{
|
|
explicit TxnLog()
|
|
: typeVarChanges(nullptr)
|
|
, typePackChanges(nullptr)
|
|
, ownedSeen()
|
|
, sharedSeen(&ownedSeen)
|
|
{
|
|
}
|
|
|
|
explicit TxnLog(TxnLog* parent)
|
|
: typeVarChanges(nullptr)
|
|
, typePackChanges(nullptr)
|
|
, parent(parent)
|
|
{
|
|
if (parent)
|
|
{
|
|
sharedSeen = parent->sharedSeen;
|
|
}
|
|
else
|
|
{
|
|
sharedSeen = &ownedSeen;
|
|
}
|
|
}
|
|
|
|
explicit TxnLog(std::vector<std::pair<TypeOrPackId, TypeOrPackId>>* sharedSeen)
|
|
: typeVarChanges(nullptr)
|
|
, typePackChanges(nullptr)
|
|
, sharedSeen(sharedSeen)
|
|
{
|
|
}
|
|
|
|
TxnLog(const TxnLog&) = delete;
|
|
TxnLog& operator=(const TxnLog&) = delete;
|
|
|
|
TxnLog(TxnLog&&) = default;
|
|
TxnLog& operator=(TxnLog&&) = default;
|
|
|
|
// Gets an empty TxnLog pointer. This is useful for constructs that
|
|
// take a TxnLog, like TypePackIterator - use the empty log if you
|
|
// don't have a TxnLog to give it.
|
|
static const TxnLog* empty();
|
|
|
|
// Joins another TxnLog onto this one. You should use std::move to avoid
|
|
// copying the rhs TxnLog.
|
|
//
|
|
// If both logs talk about the same type, pack, or table, the rhs takes
|
|
// priority.
|
|
void concat(TxnLog rhs);
|
|
void concatAsIntersections(TxnLog rhs, NotNull<TypeArena> arena);
|
|
void concatAsUnion(TxnLog rhs, NotNull<TypeArena> arena);
|
|
|
|
// Commits the TxnLog, rebinding all type pointers to their pending states.
|
|
// Clears the TxnLog afterwards.
|
|
void commit();
|
|
|
|
// Clears the TxnLog without committing any pending changes.
|
|
void clear();
|
|
|
|
// Computes an inverse of this TxnLog at the current time.
|
|
// This method should be called before commit is called in order to give an
|
|
// accurate result. Committing the inverse of a TxnLog will undo the changes
|
|
// made by commit, assuming the inverse log is accurate.
|
|
TxnLog inverse();
|
|
|
|
bool haveSeen(TypeId lhs, TypeId rhs) const;
|
|
void pushSeen(TypeId lhs, TypeId rhs);
|
|
void popSeen(TypeId lhs, TypeId rhs);
|
|
|
|
bool haveSeen(TypePackId lhs, TypePackId rhs) const;
|
|
void pushSeen(TypePackId lhs, TypePackId rhs);
|
|
void popSeen(TypePackId lhs, TypePackId rhs);
|
|
|
|
// Queues a type for modification. The original type will not change until commit
|
|
// is called. Use pending to get the pending state.
|
|
//
|
|
// The pointer returned lives until `commit` or `clear` is called.
|
|
PendingType* queue(TypeId ty);
|
|
|
|
// Queues a type pack for modification. The original type pack will not change
|
|
// until commit is called. Use pending to get the pending state.
|
|
//
|
|
// The pointer returned lives until `commit` or `clear` is called.
|
|
PendingTypePack* queue(TypePackId tp);
|
|
|
|
// Returns the pending state of a type, or nullptr if there isn't any. It is important
|
|
// to note that this pending state is not transitive: the pending state may reference
|
|
// non-pending types freely, so you may need to call pending multiple times to view the
|
|
// entire pending state of a type graph.
|
|
//
|
|
// The pointer returned lives until `commit` or `clear` is called.
|
|
PendingType* pending(TypeId ty) const;
|
|
|
|
// Returns the pending state of a type pack, or nullptr if there isn't any. It is
|
|
// important to note that this pending state is not transitive: the pending state may
|
|
// reference non-pending types freely, so you may need to call pending multiple times
|
|
// to view the entire pending state of a type graph.
|
|
//
|
|
// The pointer returned lives until `commit` or `clear` is called.
|
|
PendingTypePack* pending(TypePackId tp) const;
|
|
|
|
// Queues a replacement of a type with another type.
|
|
//
|
|
// The pointer returned lives until `commit` or `clear` is called.
|
|
PendingType* replace(TypeId ty, Type replacement);
|
|
|
|
// Queues a replacement of a type pack with another type pack.
|
|
//
|
|
// The pointer returned lives until `commit` or `clear` is called.
|
|
PendingTypePack* replace(TypePackId tp, TypePackVar replacement);
|
|
|
|
// Queues a replacement of a table type with another table type that is bound
|
|
// to a specific value.
|
|
//
|
|
// The pointer returned lives until `commit` or `clear` is called.
|
|
PendingType* bindTable(TypeId ty, std::optional<TypeId> newBoundTo);
|
|
|
|
// Queues a replacement of a type with a level with a duplicate of that type
|
|
// with a new type level.
|
|
//
|
|
// The pointer returned lives until `commit` or `clear` is called.
|
|
PendingType* changeLevel(TypeId ty, TypeLevel newLevel);
|
|
|
|
// Queues a replacement of a type pack with a level with a duplicate of that
|
|
// type pack with a new type level.
|
|
//
|
|
// The pointer returned lives until `commit` or `clear` is called.
|
|
PendingTypePack* changeLevel(TypePackId tp, TypeLevel newLevel);
|
|
|
|
// Queues a replacement of a table type with another table type with a new
|
|
// indexer.
|
|
//
|
|
// The pointer returned lives until `commit` or `clear` is called.
|
|
PendingType* changeIndexer(TypeId ty, std::optional<TableIndexer> indexer);
|
|
|
|
// Returns the type level of the pending state of the type, or the level of that
|
|
// type, if no pending state exists. If the type doesn't have a notion of a level,
|
|
// returns nullopt. If the pending state doesn't have a notion of a level, but the
|
|
// original state does, returns nullopt.
|
|
std::optional<TypeLevel> getLevel(TypeId ty) const;
|
|
|
|
// Follows a type, accounting for pending type states. The returned type may have
|
|
// pending state; you should use `pending` or `get` to find out.
|
|
TypeId follow(TypeId ty) const;
|
|
|
|
// Follows a type pack, accounting for pending type states. The returned type pack
|
|
// may have pending state; you should use `pending` or `get` to find out.
|
|
TypePackId follow(TypePackId tp) const;
|
|
|
|
// Replaces a given type's state with a new variant. Returns the new pending state
|
|
// of that type.
|
|
//
|
|
// The pointer returned lives until `commit` or `clear` is called.
|
|
template<typename T>
|
|
PendingType* replace(TypeId ty, T replacement)
|
|
{
|
|
return replace(ty, Type(replacement));
|
|
}
|
|
|
|
// Replaces a given type pack's state with a new variant. Returns the new
|
|
// pending state of that type pack.
|
|
//
|
|
// The pointer returned lives until `commit` or `clear` is called.
|
|
template<typename T>
|
|
PendingTypePack* replace(TypePackId tp, T replacement)
|
|
{
|
|
return replace(tp, TypePackVar(replacement));
|
|
}
|
|
|
|
// Returns T if a given type or type pack is this variant, respecting the
|
|
// log's pending state.
|
|
//
|
|
// Do not retain this pointer; it has the potential to be invalidated when
|
|
// commit or clear is called.
|
|
template<typename T, typename TID>
|
|
T* getMutable(TID ty) const
|
|
{
|
|
auto* pendingTy = pending(ty);
|
|
if (pendingTy)
|
|
return Luau::getMutable<T>(pendingTy);
|
|
|
|
return Luau::getMutable<T>(ty);
|
|
}
|
|
|
|
template<typename T, typename TID>
|
|
const T* get(TID ty) const
|
|
{
|
|
return this->getMutable<T>(ty);
|
|
}
|
|
|
|
// Returns whether a given type or type pack is a given state, respecting the
|
|
// log's pending state.
|
|
//
|
|
// This method will not assert if called on a BoundType or BoundTypePack.
|
|
template<typename T, typename TID>
|
|
bool is(TID ty) const
|
|
{
|
|
// We do not use getMutable here because this method can be called on
|
|
// BoundTypes, which triggers an assertion.
|
|
auto* pendingTy = pending(ty);
|
|
if (pendingTy)
|
|
return Luau::get_if<T>(&pendingTy->pending.ty) != nullptr;
|
|
|
|
return Luau::get_if<T>(&ty->ty) != nullptr;
|
|
}
|
|
|
|
std::pair<std::vector<TypeId>, std::vector<TypePackId>> getChanges() const;
|
|
|
|
private:
|
|
// unique_ptr is used to give us stable pointers across insertions into the
|
|
// map. Otherwise, it would be really easy to accidentally invalidate the
|
|
// pointers returned from queue/pending.
|
|
DenseHashMap<TypeId, std::unique_ptr<PendingType>> typeVarChanges;
|
|
DenseHashMap<TypePackId, std::unique_ptr<PendingTypePack>> typePackChanges;
|
|
|
|
TxnLog* parent = nullptr;
|
|
|
|
// Owned version of sharedSeen. This should not be accessed directly in
|
|
// TxnLogs; use sharedSeen instead. This field exists because in the tree
|
|
// of TxnLogs, the root must own its seen set. In all descendant TxnLogs,
|
|
// this is an empty vector.
|
|
std::vector<std::pair<TypeOrPackId, TypeOrPackId>> ownedSeen;
|
|
|
|
bool haveSeen(TypeOrPackId lhs, TypeOrPackId rhs) const;
|
|
void pushSeen(TypeOrPackId lhs, TypeOrPackId rhs);
|
|
void popSeen(TypeOrPackId lhs, TypeOrPackId rhs);
|
|
|
|
public:
|
|
// There is one spot in the code where TxnLog has to reconcile collisions
|
|
// between parallel logs. In that codepath, we have to work out which of two
|
|
// FreeTypes subsumes the other. If useScopes is false, the TypeLevel is
|
|
// used. Else we use the embedded Scope*.
|
|
bool useScopes = false;
|
|
|
|
// It is sometimes the case under DCR that we speculatively rebind
|
|
// GenericTypes to other types as though they were free. We mark logs that
|
|
// contain these kinds of substitutions as radioactive so that we know that
|
|
// we must never commit one.
|
|
bool radioactive = false;
|
|
|
|
// Used to avoid infinite recursion when types are cyclic.
|
|
// Shared with all the descendent TxnLogs.
|
|
std::vector<std::pair<TypeOrPackId, TypeOrPackId>>* sharedSeen;
|
|
};
|
|
|
|
} // namespace Luau
|