2023-06-16 18:01:18 +01:00
|
|
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
|
|
#include "Types.h"
|
|
|
|
|
2023-08-11 13:55:30 +01:00
|
|
|
#include "Luau/BytecodeBuilder.h"
|
|
|
|
|
2024-04-25 21:57:23 +01:00
|
|
|
LUAU_FASTFLAG(LuauCompileTypeInfo)
|
|
|
|
|
2023-06-16 18:01:18 +01:00
|
|
|
namespace Luau
|
|
|
|
{
|
|
|
|
|
2023-07-07 18:14:35 +01:00
|
|
|
static bool isGeneric(AstName name, const AstArray<AstGenericType>& generics)
|
|
|
|
{
|
|
|
|
for (const AstGenericType& gt : generics)
|
|
|
|
if (gt.name == name)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-07-14 16:57:16 +01:00
|
|
|
static LuauBytecodeType getPrimitiveType(AstName name)
|
2023-07-07 18:14:35 +01:00
|
|
|
{
|
|
|
|
if (name == "nil")
|
|
|
|
return LBC_TYPE_NIL;
|
|
|
|
else if (name == "boolean")
|
|
|
|
return LBC_TYPE_BOOLEAN;
|
|
|
|
else if (name == "number")
|
|
|
|
return LBC_TYPE_NUMBER;
|
|
|
|
else if (name == "string")
|
|
|
|
return LBC_TYPE_STRING;
|
|
|
|
else if (name == "thread")
|
|
|
|
return LBC_TYPE_THREAD;
|
2024-01-12 19:16:39 +00:00
|
|
|
else if (name == "buffer")
|
2023-10-27 20:33:36 +01:00
|
|
|
return LBC_TYPE_BUFFER;
|
2023-07-07 18:14:35 +01:00
|
|
|
else if (name == "any" || name == "unknown")
|
|
|
|
return LBC_TYPE_ANY;
|
|
|
|
else
|
|
|
|
return LBC_TYPE_INVALID;
|
|
|
|
}
|
|
|
|
|
2023-07-14 16:57:16 +01:00
|
|
|
static LuauBytecodeType getType(AstType* ty, const AstArray<AstGenericType>& generics, const DenseHashMap<AstName, AstStatTypeAlias*>& typeAliases,
|
|
|
|
bool resolveAliases, const char* vectorType)
|
2023-06-16 18:01:18 +01:00
|
|
|
{
|
|
|
|
if (AstTypeReference* ref = ty->as<AstTypeReference>())
|
|
|
|
{
|
2023-07-07 18:14:35 +01:00
|
|
|
if (ref->prefix)
|
2023-06-16 18:01:18 +01:00
|
|
|
return LBC_TYPE_ANY;
|
2023-07-07 18:14:35 +01:00
|
|
|
|
|
|
|
if (AstStatTypeAlias* const* alias = typeAliases.find(ref->name); alias && *alias)
|
|
|
|
{
|
|
|
|
// note: we only resolve aliases to the depth of 1 to avoid dealing with recursive aliases
|
|
|
|
if (resolveAliases)
|
2023-07-14 16:57:16 +01:00
|
|
|
return getType((*alias)->type, (*alias)->generics, typeAliases, /* resolveAliases= */ false, vectorType);
|
2023-07-07 18:14:35 +01:00
|
|
|
else
|
|
|
|
return LBC_TYPE_ANY;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isGeneric(ref->name, generics))
|
|
|
|
return LBC_TYPE_ANY;
|
|
|
|
|
2023-07-14 16:57:16 +01:00
|
|
|
if (vectorType && ref->name == vectorType)
|
|
|
|
return LBC_TYPE_VECTOR;
|
|
|
|
|
|
|
|
if (LuauBytecodeType prim = getPrimitiveType(ref->name); prim != LBC_TYPE_INVALID)
|
2023-07-07 18:14:35 +01:00
|
|
|
return prim;
|
|
|
|
|
|
|
|
// not primitive or alias or generic => host-provided, we assume userdata for now
|
|
|
|
return LBC_TYPE_USERDATA;
|
2023-06-16 18:01:18 +01:00
|
|
|
}
|
|
|
|
else if (AstTypeTable* table = ty->as<AstTypeTable>())
|
|
|
|
{
|
|
|
|
return LBC_TYPE_TABLE;
|
|
|
|
}
|
|
|
|
else if (AstTypeFunction* func = ty->as<AstTypeFunction>())
|
|
|
|
{
|
|
|
|
return LBC_TYPE_FUNCTION;
|
|
|
|
}
|
|
|
|
else if (AstTypeUnion* un = ty->as<AstTypeUnion>())
|
|
|
|
{
|
|
|
|
bool optional = false;
|
2023-07-14 16:57:16 +01:00
|
|
|
LuauBytecodeType type = LBC_TYPE_INVALID;
|
2023-06-16 18:01:18 +01:00
|
|
|
|
|
|
|
for (AstType* ty : un->types)
|
|
|
|
{
|
2023-07-14 16:57:16 +01:00
|
|
|
LuauBytecodeType et = getType(ty, generics, typeAliases, resolveAliases, vectorType);
|
2023-06-16 18:01:18 +01:00
|
|
|
|
|
|
|
if (et == LBC_TYPE_NIL)
|
|
|
|
{
|
|
|
|
optional = true;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (type == LBC_TYPE_INVALID)
|
|
|
|
{
|
|
|
|
type = et;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (type != et)
|
|
|
|
return LBC_TYPE_ANY;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (type == LBC_TYPE_INVALID)
|
|
|
|
return LBC_TYPE_ANY;
|
|
|
|
|
2023-07-14 16:57:16 +01:00
|
|
|
return LuauBytecodeType(type | (optional && (type != LBC_TYPE_ANY) ? LBC_TYPE_OPTIONAL_BIT : 0));
|
2023-06-16 18:01:18 +01:00
|
|
|
}
|
|
|
|
else if (AstTypeIntersection* inter = ty->as<AstTypeIntersection>())
|
|
|
|
{
|
|
|
|
return LBC_TYPE_ANY;
|
|
|
|
}
|
|
|
|
|
|
|
|
return LBC_TYPE_ANY;
|
|
|
|
}
|
|
|
|
|
2023-07-14 16:57:16 +01:00
|
|
|
static std::string getFunctionType(const AstExprFunction* func, const DenseHashMap<AstName, AstStatTypeAlias*>& typeAliases, const char* vectorType)
|
2023-06-16 18:01:18 +01:00
|
|
|
{
|
|
|
|
bool self = func->self != 0;
|
|
|
|
|
|
|
|
std::string typeInfo;
|
|
|
|
typeInfo.reserve(func->args.size + self + 2);
|
|
|
|
|
|
|
|
typeInfo.push_back(LBC_TYPE_FUNCTION);
|
|
|
|
typeInfo.push_back(uint8_t(self + func->args.size));
|
|
|
|
|
|
|
|
if (self)
|
|
|
|
typeInfo.push_back(LBC_TYPE_TABLE);
|
|
|
|
|
|
|
|
bool haveNonAnyParam = false;
|
|
|
|
for (AstLocal* arg : func->args)
|
|
|
|
{
|
2023-07-14 16:57:16 +01:00
|
|
|
LuauBytecodeType ty =
|
|
|
|
arg->annotation ? getType(arg->annotation, func->generics, typeAliases, /* resolveAliases= */ true, vectorType) : LBC_TYPE_ANY;
|
2023-06-16 18:01:18 +01:00
|
|
|
|
|
|
|
if (ty != LBC_TYPE_ANY)
|
|
|
|
haveNonAnyParam = true;
|
|
|
|
|
|
|
|
typeInfo.push_back(ty);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If all parameters simplify to any, we can just omit type info for this function
|
|
|
|
if (!haveNonAnyParam)
|
|
|
|
return {};
|
|
|
|
|
|
|
|
return typeInfo;
|
|
|
|
}
|
|
|
|
|
2023-07-07 18:14:35 +01:00
|
|
|
struct TypeMapVisitor : AstVisitor
|
|
|
|
{
|
2024-04-25 21:57:23 +01:00
|
|
|
DenseHashMap<AstExprFunction*, std::string>& functionTypes;
|
|
|
|
DenseHashMap<AstLocal*, LuauBytecodeType>& localTypes;
|
2023-07-14 16:57:16 +01:00
|
|
|
const char* vectorType;
|
2023-07-07 18:14:35 +01:00
|
|
|
|
|
|
|
DenseHashMap<AstName, AstStatTypeAlias*> typeAliases;
|
|
|
|
std::vector<std::pair<AstName, AstStatTypeAlias*>> typeAliasStack;
|
|
|
|
|
2024-04-25 21:57:23 +01:00
|
|
|
TypeMapVisitor(
|
|
|
|
DenseHashMap<AstExprFunction*, std::string>& functionTypes, DenseHashMap<AstLocal*, LuauBytecodeType>& localTypes, const char* vectorType)
|
|
|
|
: functionTypes(functionTypes)
|
|
|
|
, localTypes(localTypes)
|
2023-07-14 16:57:16 +01:00
|
|
|
, vectorType(vectorType)
|
2023-07-07 18:14:35 +01:00
|
|
|
, typeAliases(AstName())
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t pushTypeAliases(AstStatBlock* block)
|
|
|
|
{
|
|
|
|
size_t aliasStackTop = typeAliasStack.size();
|
|
|
|
|
|
|
|
for (AstStat* stat : block->body)
|
|
|
|
if (AstStatTypeAlias* alias = stat->as<AstStatTypeAlias>())
|
|
|
|
{
|
|
|
|
AstStatTypeAlias*& prevAlias = typeAliases[alias->name];
|
|
|
|
|
|
|
|
typeAliasStack.push_back(std::make_pair(alias->name, prevAlias));
|
|
|
|
prevAlias = alias;
|
|
|
|
}
|
|
|
|
|
|
|
|
return aliasStackTop;
|
|
|
|
}
|
|
|
|
|
|
|
|
void popTypeAliases(size_t aliasStackTop)
|
|
|
|
{
|
|
|
|
while (typeAliasStack.size() > aliasStackTop)
|
|
|
|
{
|
|
|
|
std::pair<AstName, AstStatTypeAlias*>& top = typeAliasStack.back();
|
|
|
|
|
|
|
|
typeAliases[top.first] = top.second;
|
|
|
|
typeAliasStack.pop_back();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(AstStatBlock* node) override
|
|
|
|
{
|
|
|
|
size_t aliasStackTop = pushTypeAliases(node);
|
|
|
|
|
|
|
|
for (AstStat* stat : node->body)
|
|
|
|
stat->visit(this);
|
|
|
|
|
|
|
|
popTypeAliases(aliasStackTop);
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// repeat..until scoping rules are such that condition (along with any possible functions declared in it) has aliases from repeat body in scope
|
|
|
|
bool visit(AstStatRepeat* node) override
|
|
|
|
{
|
|
|
|
size_t aliasStackTop = pushTypeAliases(node->body);
|
|
|
|
|
|
|
|
for (AstStat* stat : node->body->body)
|
|
|
|
stat->visit(this);
|
|
|
|
|
|
|
|
node->condition->visit(this);
|
|
|
|
|
|
|
|
popTypeAliases(aliasStackTop);
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(AstExprFunction* node) override
|
|
|
|
{
|
2023-07-14 16:57:16 +01:00
|
|
|
std::string type = getFunctionType(node, typeAliases, vectorType);
|
2023-07-07 18:14:35 +01:00
|
|
|
|
|
|
|
if (!type.empty())
|
2024-04-25 21:57:23 +01:00
|
|
|
functionTypes[node] = std::move(type);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(AstExprLocal* node) override
|
|
|
|
{
|
|
|
|
if (FFlag::LuauCompileTypeInfo)
|
|
|
|
{
|
|
|
|
AstLocal* local = node->local;
|
|
|
|
|
|
|
|
if (AstType* annotation = local->annotation)
|
|
|
|
{
|
|
|
|
LuauBytecodeType ty = getType(annotation, {}, typeAliases, /* resolveAliases= */ true, vectorType);
|
|
|
|
|
|
|
|
if (ty != LBC_TYPE_ANY)
|
|
|
|
localTypes[local] = ty;
|
|
|
|
}
|
|
|
|
}
|
2023-07-07 18:14:35 +01:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2024-04-25 21:57:23 +01:00
|
|
|
void buildTypeMap(DenseHashMap<AstExprFunction*, std::string>& functionTypes, DenseHashMap<AstLocal*, LuauBytecodeType>& localTypes, AstNode* root,
|
|
|
|
const char* vectorType)
|
2023-07-07 18:14:35 +01:00
|
|
|
{
|
2024-04-25 21:57:23 +01:00
|
|
|
TypeMapVisitor visitor(functionTypes, localTypes, vectorType);
|
2023-07-07 18:14:35 +01:00
|
|
|
root->visit(&visitor);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace Luau
|