mirror of
https://github.com/luau-lang/luau.git
synced 2025-08-26 11:27:08 +01:00
As always, a weekly Luau update! This week we have further improvements to new type solver, fixing a few of the popular issues reported. The fragment autocomplete is even more stable and we believe it's ready for broader use. Aside from that we have a few general fixes/improvements: * Fixed data race when multi-threaded typechecking is used, appearing as a random crash at the end of typechecking * AST data is now available from `Luau::Module` ## New Type Solver * Fixed type refinements made by function calls which could attach `nil` as an option of a type before (Fixes #1528) * Improved bidirectional typechecking in tables (Fixes #1596) * Fixed normalization of negated types * `getmetatable()` on `any` type should no longer report an error ## Fragment Autocomplete * Fixed auto-complete suggestions being provided inside multiline comments * Fixed an assertion failure that could happen when old type solver was used * Fixed issues with missing suggestions when multiple statements are on the same line * Fixed memory safety issues ## Internal Contributors Co-authored-by: Andy Friesen <afriesen@roblox.com> Co-authored-by: Hunter Goldstein <hgoldstein@roblox.com> Co-authored-by: Vighnesh Vijay <vvijay@roblox.com> Co-authored-by: Vyacheslav Egorov <vegorov@roblox.com>
68 lines
1.7 KiB
C++
68 lines
1.7 KiB
C++
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
#include "Luau/Refinement.h"
|
|
#include <algorithm>
|
|
|
|
namespace Luau
|
|
{
|
|
|
|
RefinementId RefinementArena::variadic(const std::vector<RefinementId>& refis)
|
|
{
|
|
bool hasRefinements = false;
|
|
for (RefinementId r : refis)
|
|
hasRefinements |= bool(r);
|
|
|
|
if (!hasRefinements)
|
|
return nullptr;
|
|
|
|
return NotNull{allocator.allocate(Variadic{refis})};
|
|
}
|
|
|
|
RefinementId RefinementArena::negation(RefinementId refinement)
|
|
{
|
|
if (!refinement)
|
|
return nullptr;
|
|
|
|
return NotNull{allocator.allocate(Negation{refinement})};
|
|
}
|
|
|
|
RefinementId RefinementArena::conjunction(RefinementId lhs, RefinementId rhs)
|
|
{
|
|
if (!lhs && !rhs)
|
|
return nullptr;
|
|
|
|
return NotNull{allocator.allocate(Conjunction{lhs, rhs})};
|
|
}
|
|
|
|
RefinementId RefinementArena::disjunction(RefinementId lhs, RefinementId rhs)
|
|
{
|
|
if (!lhs && !rhs)
|
|
return nullptr;
|
|
|
|
return NotNull{allocator.allocate(Disjunction{lhs, rhs})};
|
|
}
|
|
|
|
RefinementId RefinementArena::equivalence(RefinementId lhs, RefinementId rhs)
|
|
{
|
|
if (!lhs && !rhs)
|
|
return nullptr;
|
|
|
|
return NotNull{allocator.allocate(Equivalence{lhs, rhs})};
|
|
}
|
|
|
|
RefinementId RefinementArena::proposition(const RefinementKey* key, TypeId discriminantTy)
|
|
{
|
|
if (!key)
|
|
return nullptr;
|
|
|
|
return NotNull{allocator.allocate(Proposition{key, discriminantTy, false})};
|
|
}
|
|
|
|
RefinementId RefinementArena::implicitProposition(const RefinementKey* key, TypeId discriminantTy)
|
|
{
|
|
if (!key)
|
|
return nullptr;
|
|
|
|
return NotNull{allocator.allocate(Proposition{key, discriminantTy, true})};
|
|
}
|
|
|
|
} // namespace Luau
|