mirror of
https://github.com/luau-lang/luau.git
synced 2025-04-05 11:20:54 +01:00
# What's changed? Add program argument passing to scripts run using the Luau REPL! You can now pass `--program-args` (or shorthand `-a`) to the REPL which will treat all remaining arguments as arguments to pass to executed scripts. These values can be accessed through variadic argument expansion. You can read these values like so: ``` local args = {...} -- gets you an array of all the arguments ``` For example if we run the following script like `luau test.lua -a test1 test2 test3`: ``` -- test.lua print(...) ``` you should get the output: ``` test1 test2 test3 ``` ### Native Code Generation * Improve A64 lowering for vector operations by using vector instructions * Fix lowering issue in IR value location tracking! - A developer reported a divergence between code run in the VM and Native Code Generation which we have now fixed ### New Type Solver * Apply substitution to type families, and emit new constraints to reduce those further * More progress on reducing comparison (`lt/le`)type families * Resolve two major sources of cyclic types in the new solver ### Miscellaneous * Turned internal compiler errors (ICE's) into warnings and errors ------- Co-authored-by: Aaron Weiss <aaronweiss@roblox.com> Co-authored-by: Alexander McCord <amccord@roblox.com> Co-authored-by: Andy Friesen <afriesen@roblox.com> Co-authored-by: Aviral Goel <agoel@roblox.com> Co-authored-by: Vyacheslav Egorov <vegorov@roblox.com> --------- Co-authored-by: Aaron Weiss <aaronweiss@roblox.com> Co-authored-by: Alexander McCord <amccord@roblox.com> Co-authored-by: Andy Friesen <afriesen@roblox.com> Co-authored-by: Aviral Goel <agoel@roblox.com> Co-authored-by: David Cope <dcope@roblox.com> Co-authored-by: Lily Brown <lbrown@roblox.com> Co-authored-by: Vyacheslav Egorov <vegorov@roblox.com>
186 lines
5.4 KiB
C++
186 lines
5.4 KiB
C++
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
#include "Luau/Instantiation.h"
|
|
|
|
#include "Luau/Common.h"
|
|
#include "Luau/Instantiation2.h" // including for `Replacer` which was stolen since it will be kept in the new solver
|
|
#include "Luau/ToString.h"
|
|
#include "Luau/TxnLog.h"
|
|
#include "Luau/TypeArena.h"
|
|
#include "Luau/TypeCheckLimits.h"
|
|
|
|
#include <algorithm>
|
|
|
|
LUAU_FASTFLAG(DebugLuauDeferredConstraintResolution)
|
|
|
|
namespace Luau
|
|
{
|
|
|
|
bool Instantiation::isDirty(TypeId ty)
|
|
{
|
|
if (const FunctionType* ftv = log->getMutable<FunctionType>(ty))
|
|
{
|
|
if (ftv->hasNoFreeOrGenericTypes)
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
bool Instantiation::isDirty(TypePackId tp)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
bool Instantiation::ignoreChildren(TypeId ty)
|
|
{
|
|
if (log->getMutable<FunctionType>(ty))
|
|
return true;
|
|
else if (get<ClassType>(ty))
|
|
return true;
|
|
else
|
|
return false;
|
|
}
|
|
|
|
TypeId Instantiation::clean(TypeId ty)
|
|
{
|
|
const FunctionType* ftv = log->getMutable<FunctionType>(ty);
|
|
LUAU_ASSERT(ftv);
|
|
|
|
FunctionType clone = FunctionType{level, scope, ftv->argTypes, ftv->retTypes, ftv->definition, ftv->hasSelf};
|
|
clone.magicFunction = ftv->magicFunction;
|
|
clone.dcrMagicFunction = ftv->dcrMagicFunction;
|
|
clone.dcrMagicRefinement = ftv->dcrMagicRefinement;
|
|
clone.tags = ftv->tags;
|
|
clone.argNames = ftv->argNames;
|
|
TypeId result = addType(std::move(clone));
|
|
|
|
// Annoyingly, we have to do this even if there are no generics,
|
|
// to replace any generic tables.
|
|
ReplaceGenerics replaceGenerics{log, arena, builtinTypes, level, scope, ftv->generics, ftv->genericPacks};
|
|
|
|
// TODO: What to do if this returns nullopt?
|
|
// We don't have access to the error-reporting machinery
|
|
result = replaceGenerics.substitute(result).value_or(result);
|
|
|
|
asMutable(result)->documentationSymbol = ty->documentationSymbol;
|
|
return result;
|
|
}
|
|
|
|
TypePackId Instantiation::clean(TypePackId tp)
|
|
{
|
|
LUAU_ASSERT(false);
|
|
return tp;
|
|
}
|
|
|
|
bool ReplaceGenerics::ignoreChildren(TypeId ty)
|
|
{
|
|
if (const FunctionType* ftv = log->getMutable<FunctionType>(ty))
|
|
{
|
|
if (ftv->hasNoFreeOrGenericTypes)
|
|
return true;
|
|
|
|
// We aren't recursing in the case of a generic function which
|
|
// binds the same generics. This can happen if, for example, there's recursive types.
|
|
// If T = <a>(a,T)->T then instantiating T should produce T' = (X,T)->T not T' = (X,T')->T'.
|
|
// It's OK to use vector equality here, since we always generate fresh generics
|
|
// whenever we quantify, so the vectors overlap if and only if they are equal.
|
|
return (!generics.empty() || !genericPacks.empty()) && (ftv->generics == generics) && (ftv->genericPacks == genericPacks);
|
|
}
|
|
else if (get<ClassType>(ty))
|
|
return true;
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
bool ReplaceGenerics::isDirty(TypeId ty)
|
|
{
|
|
if (const TableType* ttv = log->getMutable<TableType>(ty))
|
|
return ttv->state == TableState::Generic;
|
|
else if (log->getMutable<GenericType>(ty))
|
|
return std::find(generics.begin(), generics.end(), ty) != generics.end();
|
|
else
|
|
return false;
|
|
}
|
|
|
|
bool ReplaceGenerics::isDirty(TypePackId tp)
|
|
{
|
|
if (log->getMutable<GenericTypePack>(tp))
|
|
return std::find(genericPacks.begin(), genericPacks.end(), tp) != genericPacks.end();
|
|
else
|
|
return false;
|
|
}
|
|
|
|
TypeId ReplaceGenerics::clean(TypeId ty)
|
|
{
|
|
LUAU_ASSERT(isDirty(ty));
|
|
if (const TableType* ttv = log->getMutable<TableType>(ty))
|
|
{
|
|
TableType clone = TableType{ttv->props, ttv->indexer, level, scope, TableState::Free};
|
|
clone.definitionModuleName = ttv->definitionModuleName;
|
|
clone.definitionLocation = ttv->definitionLocation;
|
|
return addType(std::move(clone));
|
|
}
|
|
else if (FFlag::DebugLuauDeferredConstraintResolution)
|
|
{
|
|
TypeId res = freshType(NotNull{arena}, builtinTypes, scope);
|
|
getMutable<FreeType>(res)->level = level;
|
|
return res;
|
|
}
|
|
else
|
|
{
|
|
return addType(FreeType{scope, level});
|
|
}
|
|
}
|
|
|
|
TypePackId ReplaceGenerics::clean(TypePackId tp)
|
|
{
|
|
LUAU_ASSERT(isDirty(tp));
|
|
return addTypePack(TypePackVar(FreeTypePack{scope, level}));
|
|
}
|
|
|
|
std::optional<TypeId> instantiate(
|
|
NotNull<BuiltinTypes> builtinTypes, NotNull<TypeArena> arena, NotNull<TypeCheckLimits> limits, NotNull<Scope> scope, TypeId ty)
|
|
{
|
|
ty = follow(ty);
|
|
|
|
const FunctionType* ft = get<FunctionType>(ty);
|
|
if (!ft)
|
|
return ty;
|
|
|
|
if (ft->generics.empty() && ft->genericPacks.empty())
|
|
return ty;
|
|
|
|
DenseHashMap<TypeId, TypeId> replacements{nullptr};
|
|
DenseHashMap<TypePackId, TypePackId> replacementPacks{nullptr};
|
|
|
|
for (TypeId g : ft->generics)
|
|
replacements[g] = freshType(arena, builtinTypes, scope);
|
|
|
|
for (TypePackId g : ft->genericPacks)
|
|
replacementPacks[g] = arena->freshTypePack(scope);
|
|
|
|
Replacer r{arena, std::move(replacements), std::move(replacementPacks)};
|
|
|
|
if (limits->instantiationChildLimit)
|
|
r.childLimit = *limits->instantiationChildLimit;
|
|
|
|
std::optional<TypeId> res = r.substitute(ty);
|
|
if (!res)
|
|
return res;
|
|
|
|
FunctionType* ft2 = getMutable<FunctionType>(*res);
|
|
LUAU_ASSERT(ft != ft2);
|
|
|
|
ft2->generics.clear();
|
|
ft2->genericPacks.clear();
|
|
|
|
return res;
|
|
}
|
|
|
|
} // namespace Luau
|