2022-01-14 16:20:09 +00:00
|
|
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
|
|
#include "TableShape.h"
|
|
|
|
|
|
|
|
namespace Luau
|
|
|
|
{
|
|
|
|
namespace Compile
|
|
|
|
{
|
|
|
|
|
2022-01-21 17:00:19 +00:00
|
|
|
// conservative limit for the loop bound that establishes table array size
|
|
|
|
static const int kMaxLoopBound = 16;
|
|
|
|
|
2022-01-14 16:20:09 +00:00
|
|
|
static AstExprTable* getTableHint(AstExpr* expr)
|
|
|
|
{
|
|
|
|
// unadorned table literal
|
|
|
|
if (AstExprTable* table = expr->as<AstExprTable>())
|
|
|
|
return table;
|
|
|
|
|
|
|
|
// setmetatable(table literal, ...)
|
|
|
|
if (AstExprCall* call = expr->as<AstExprCall>(); call && !call->self && call->args.size == 2)
|
|
|
|
if (AstExprGlobal* func = call->func->as<AstExprGlobal>(); func && func->name == "setmetatable")
|
|
|
|
if (AstExprTable* table = call->args.data[0]->as<AstExprTable>())
|
|
|
|
return table;
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct ShapeVisitor : AstVisitor
|
|
|
|
{
|
|
|
|
struct Hasher
|
|
|
|
{
|
|
|
|
size_t operator()(const std::pair<AstExprTable*, AstName>& p) const
|
|
|
|
{
|
2022-01-21 17:00:19 +00:00
|
|
|
return DenseHashPointer()(p.first) ^ std::hash<AstName>()(p.second);
|
2022-01-14 16:20:09 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
DenseHashMap<AstExprTable*, TableShape>& shapes;
|
|
|
|
|
|
|
|
DenseHashMap<AstLocal*, AstExprTable*> tables;
|
|
|
|
DenseHashSet<std::pair<AstExprTable*, AstName>, Hasher> fields;
|
|
|
|
|
2022-01-21 17:00:19 +00:00
|
|
|
DenseHashMap<AstLocal*, unsigned int> loops; // iterator => upper bound for 1..k
|
|
|
|
|
2022-01-14 16:20:09 +00:00
|
|
|
ShapeVisitor(DenseHashMap<AstExprTable*, TableShape>& shapes)
|
|
|
|
: shapes(shapes)
|
|
|
|
, tables(nullptr)
|
|
|
|
, fields(std::pair<AstExprTable*, AstName>())
|
2022-01-21 17:00:19 +00:00
|
|
|
, loops(nullptr)
|
2022-01-14 16:20:09 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void assignField(AstExpr* expr, AstName index)
|
|
|
|
{
|
|
|
|
if (AstExprLocal* lv = expr->as<AstExprLocal>())
|
|
|
|
{
|
|
|
|
if (AstExprTable** table = tables.find(lv->local))
|
|
|
|
{
|
|
|
|
std::pair<AstExprTable*, AstName> field = {*table, index};
|
|
|
|
|
|
|
|
if (!fields.contains(field))
|
|
|
|
{
|
|
|
|
fields.insert(field);
|
|
|
|
shapes[*table].hashSize += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void assignField(AstExpr* expr, AstExpr* index)
|
|
|
|
{
|
|
|
|
AstExprLocal* lv = expr->as<AstExprLocal>();
|
2022-01-21 17:00:19 +00:00
|
|
|
if (!lv)
|
|
|
|
return;
|
|
|
|
|
|
|
|
AstExprTable** table = tables.find(lv->local);
|
|
|
|
if (!table)
|
|
|
|
return;
|
2022-01-14 16:20:09 +00:00
|
|
|
|
2022-01-21 17:00:19 +00:00
|
|
|
if (AstExprConstantNumber* number = index->as<AstExprConstantNumber>())
|
2022-01-14 16:20:09 +00:00
|
|
|
{
|
2022-01-21 17:00:19 +00:00
|
|
|
TableShape& shape = shapes[*table];
|
|
|
|
|
|
|
|
if (number->value == double(shape.arraySize + 1))
|
|
|
|
shape.arraySize += 1;
|
|
|
|
}
|
|
|
|
else if (AstExprLocal* iter = index->as<AstExprLocal>())
|
|
|
|
{
|
|
|
|
if (const unsigned int* bound = loops.find(iter->local))
|
2022-01-14 16:20:09 +00:00
|
|
|
{
|
|
|
|
TableShape& shape = shapes[*table];
|
|
|
|
|
2022-01-21 17:00:19 +00:00
|
|
|
if (shape.arraySize == 0)
|
|
|
|
shape.arraySize = *bound;
|
2022-01-14 16:20:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void assign(AstExpr* var)
|
|
|
|
{
|
|
|
|
if (AstExprIndexName* index = var->as<AstExprIndexName>())
|
|
|
|
{
|
|
|
|
assignField(index->expr, index->index);
|
|
|
|
}
|
|
|
|
else if (AstExprIndexExpr* index = var->as<AstExprIndexExpr>())
|
|
|
|
{
|
|
|
|
assignField(index->expr, index->index);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(AstStatLocal* node) override
|
|
|
|
{
|
|
|
|
// track local -> table association so that we can update table size prediction in assignField
|
|
|
|
if (node->vars.size == 1 && node->values.size == 1)
|
|
|
|
if (AstExprTable* table = getTableHint(node->values.data[0]); table && table->items.size == 0)
|
|
|
|
tables[node->vars.data[0]] = table;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(AstStatAssign* node) override
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < node->vars.size; ++i)
|
|
|
|
assign(node->vars.data[i]);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < node->values.size; ++i)
|
|
|
|
node->values.data[i]->visit(this);
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(AstStatFunction* node) override
|
|
|
|
{
|
|
|
|
assign(node->name);
|
|
|
|
node->func->visit(this);
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
2022-01-21 17:00:19 +00:00
|
|
|
|
|
|
|
bool visit(AstStatFor* node) override
|
|
|
|
{
|
|
|
|
AstExprConstantNumber* from = node->from->as<AstExprConstantNumber>();
|
|
|
|
AstExprConstantNumber* to = node->to->as<AstExprConstantNumber>();
|
|
|
|
|
|
|
|
if (from && to && from->value == 1.0 && to->value >= 1.0 && to->value <= double(kMaxLoopBound) && !node->step)
|
|
|
|
loops[node->var] = unsigned(to->value);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2022-01-14 16:20:09 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
void predictTableShapes(DenseHashMap<AstExprTable*, TableShape>& shapes, AstNode* root)
|
|
|
|
{
|
|
|
|
ShapeVisitor visitor{shapes};
|
|
|
|
root->visit(&visitor);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace Compile
|
|
|
|
} // namespace Luau
|