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 "Builtins.h"
|
|
|
|
|
|
|
|
#include "Luau/Bytecode.h"
|
|
|
|
#include "Luau/Compiler.h"
|
|
|
|
|
2022-09-29 23:23:10 +01:00
|
|
|
LUAU_FASTFLAGVARIABLE(LuauCompileBuiltinMT, false)
|
|
|
|
|
2022-01-14 16:20:09 +00:00
|
|
|
namespace Luau
|
|
|
|
{
|
|
|
|
namespace Compile
|
|
|
|
{
|
|
|
|
|
|
|
|
Builtin getBuiltin(AstExpr* node, const DenseHashMap<AstName, Global>& globals, const DenseHashMap<AstLocal*, Variable>& variables)
|
|
|
|
{
|
|
|
|
if (AstExprLocal* expr = node->as<AstExprLocal>())
|
|
|
|
{
|
|
|
|
const Variable* v = variables.find(expr->local);
|
|
|
|
|
|
|
|
return v && !v->written && v->init ? getBuiltin(v->init, globals, variables) : Builtin();
|
|
|
|
}
|
|
|
|
else if (AstExprIndexName* expr = node->as<AstExprIndexName>())
|
|
|
|
{
|
|
|
|
if (AstExprGlobal* object = expr->expr->as<AstExprGlobal>())
|
|
|
|
{
|
|
|
|
return getGlobalState(globals, object->name) == Global::Default ? Builtin{object->name, expr->index} : Builtin();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return Builtin();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (AstExprGlobal* expr = node->as<AstExprGlobal>())
|
|
|
|
{
|
|
|
|
return getGlobalState(globals, expr->name) == Global::Default ? Builtin{AstName(), expr->name} : Builtin();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return Builtin();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-14 23:52:26 +01:00
|
|
|
static int getBuiltinFunctionId(const Builtin& builtin, const CompileOptions& options)
|
2022-01-14 16:20:09 +00:00
|
|
|
{
|
|
|
|
if (builtin.isGlobal("assert"))
|
|
|
|
return LBF_ASSERT;
|
|
|
|
|
|
|
|
if (builtin.isGlobal("type"))
|
|
|
|
return LBF_TYPE;
|
|
|
|
|
|
|
|
if (builtin.isGlobal("typeof"))
|
|
|
|
return LBF_TYPEOF;
|
|
|
|
|
|
|
|
if (builtin.isGlobal("rawset"))
|
|
|
|
return LBF_RAWSET;
|
|
|
|
if (builtin.isGlobal("rawget"))
|
|
|
|
return LBF_RAWGET;
|
|
|
|
if (builtin.isGlobal("rawequal"))
|
|
|
|
return LBF_RAWEQUAL;
|
2022-08-04 23:35:33 +01:00
|
|
|
if (builtin.isGlobal("rawlen"))
|
2022-07-01 00:52:43 +01:00
|
|
|
return LBF_RAWLEN;
|
2022-01-14 16:20:09 +00:00
|
|
|
|
|
|
|
if (builtin.isGlobal("unpack"))
|
|
|
|
return LBF_TABLE_UNPACK;
|
|
|
|
|
2022-03-18 00:46:04 +00:00
|
|
|
if (builtin.isGlobal("select"))
|
2022-01-27 23:46:05 +00:00
|
|
|
return LBF_SELECT_VARARG;
|
|
|
|
|
2022-09-29 23:23:10 +01:00
|
|
|
if (FFlag::LuauCompileBuiltinMT)
|
|
|
|
{
|
|
|
|
if (builtin.isGlobal("getmetatable"))
|
|
|
|
return LBF_GETMETATABLE;
|
|
|
|
if (builtin.isGlobal("setmetatable"))
|
|
|
|
return LBF_SETMETATABLE;
|
|
|
|
}
|
|
|
|
|
2022-01-14 16:20:09 +00:00
|
|
|
if (builtin.object == "math")
|
|
|
|
{
|
|
|
|
if (builtin.method == "abs")
|
|
|
|
return LBF_MATH_ABS;
|
|
|
|
if (builtin.method == "acos")
|
|
|
|
return LBF_MATH_ACOS;
|
|
|
|
if (builtin.method == "asin")
|
|
|
|
return LBF_MATH_ASIN;
|
|
|
|
if (builtin.method == "atan2")
|
|
|
|
return LBF_MATH_ATAN2;
|
|
|
|
if (builtin.method == "atan")
|
|
|
|
return LBF_MATH_ATAN;
|
|
|
|
if (builtin.method == "ceil")
|
|
|
|
return LBF_MATH_CEIL;
|
|
|
|
if (builtin.method == "cosh")
|
|
|
|
return LBF_MATH_COSH;
|
|
|
|
if (builtin.method == "cos")
|
|
|
|
return LBF_MATH_COS;
|
|
|
|
if (builtin.method == "deg")
|
|
|
|
return LBF_MATH_DEG;
|
|
|
|
if (builtin.method == "exp")
|
|
|
|
return LBF_MATH_EXP;
|
|
|
|
if (builtin.method == "floor")
|
|
|
|
return LBF_MATH_FLOOR;
|
|
|
|
if (builtin.method == "fmod")
|
|
|
|
return LBF_MATH_FMOD;
|
|
|
|
if (builtin.method == "frexp")
|
|
|
|
return LBF_MATH_FREXP;
|
|
|
|
if (builtin.method == "ldexp")
|
|
|
|
return LBF_MATH_LDEXP;
|
|
|
|
if (builtin.method == "log10")
|
|
|
|
return LBF_MATH_LOG10;
|
|
|
|
if (builtin.method == "log")
|
|
|
|
return LBF_MATH_LOG;
|
|
|
|
if (builtin.method == "max")
|
|
|
|
return LBF_MATH_MAX;
|
|
|
|
if (builtin.method == "min")
|
|
|
|
return LBF_MATH_MIN;
|
|
|
|
if (builtin.method == "modf")
|
|
|
|
return LBF_MATH_MODF;
|
|
|
|
if (builtin.method == "pow")
|
|
|
|
return LBF_MATH_POW;
|
|
|
|
if (builtin.method == "rad")
|
|
|
|
return LBF_MATH_RAD;
|
|
|
|
if (builtin.method == "sinh")
|
|
|
|
return LBF_MATH_SINH;
|
|
|
|
if (builtin.method == "sin")
|
|
|
|
return LBF_MATH_SIN;
|
|
|
|
if (builtin.method == "sqrt")
|
|
|
|
return LBF_MATH_SQRT;
|
|
|
|
if (builtin.method == "tanh")
|
|
|
|
return LBF_MATH_TANH;
|
|
|
|
if (builtin.method == "tan")
|
|
|
|
return LBF_MATH_TAN;
|
|
|
|
if (builtin.method == "clamp")
|
|
|
|
return LBF_MATH_CLAMP;
|
|
|
|
if (builtin.method == "sign")
|
|
|
|
return LBF_MATH_SIGN;
|
|
|
|
if (builtin.method == "round")
|
|
|
|
return LBF_MATH_ROUND;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (builtin.object == "bit32")
|
|
|
|
{
|
|
|
|
if (builtin.method == "arshift")
|
|
|
|
return LBF_BIT32_ARSHIFT;
|
|
|
|
if (builtin.method == "band")
|
|
|
|
return LBF_BIT32_BAND;
|
|
|
|
if (builtin.method == "bnot")
|
|
|
|
return LBF_BIT32_BNOT;
|
|
|
|
if (builtin.method == "bor")
|
|
|
|
return LBF_BIT32_BOR;
|
|
|
|
if (builtin.method == "bxor")
|
|
|
|
return LBF_BIT32_BXOR;
|
|
|
|
if (builtin.method == "btest")
|
|
|
|
return LBF_BIT32_BTEST;
|
|
|
|
if (builtin.method == "extract")
|
|
|
|
return LBF_BIT32_EXTRACT;
|
|
|
|
if (builtin.method == "lrotate")
|
|
|
|
return LBF_BIT32_LROTATE;
|
|
|
|
if (builtin.method == "lshift")
|
|
|
|
return LBF_BIT32_LSHIFT;
|
|
|
|
if (builtin.method == "replace")
|
|
|
|
return LBF_BIT32_REPLACE;
|
|
|
|
if (builtin.method == "rrotate")
|
|
|
|
return LBF_BIT32_RROTATE;
|
|
|
|
if (builtin.method == "rshift")
|
|
|
|
return LBF_BIT32_RSHIFT;
|
|
|
|
if (builtin.method == "countlz")
|
|
|
|
return LBF_BIT32_COUNTLZ;
|
|
|
|
if (builtin.method == "countrz")
|
|
|
|
return LBF_BIT32_COUNTRZ;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (builtin.object == "string")
|
|
|
|
{
|
|
|
|
if (builtin.method == "byte")
|
|
|
|
return LBF_STRING_BYTE;
|
|
|
|
if (builtin.method == "char")
|
|
|
|
return LBF_STRING_CHAR;
|
|
|
|
if (builtin.method == "len")
|
|
|
|
return LBF_STRING_LEN;
|
|
|
|
if (builtin.method == "sub")
|
|
|
|
return LBF_STRING_SUB;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (builtin.object == "table")
|
|
|
|
{
|
|
|
|
if (builtin.method == "insert")
|
|
|
|
return LBF_TABLE_INSERT;
|
|
|
|
if (builtin.method == "unpack")
|
|
|
|
return LBF_TABLE_UNPACK;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (options.vectorCtor)
|
|
|
|
{
|
|
|
|
if (options.vectorLib)
|
|
|
|
{
|
|
|
|
if (builtin.isMethod(options.vectorLib, options.vectorCtor))
|
|
|
|
return LBF_VECTOR;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (builtin.isGlobal(options.vectorCtor))
|
|
|
|
return LBF_VECTOR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2022-07-14 23:52:26 +01:00
|
|
|
struct BuiltinVisitor : AstVisitor
|
|
|
|
{
|
|
|
|
DenseHashMap<AstExprCall*, int>& result;
|
|
|
|
|
|
|
|
const DenseHashMap<AstName, Global>& globals;
|
|
|
|
const DenseHashMap<AstLocal*, Variable>& variables;
|
|
|
|
|
|
|
|
const CompileOptions& options;
|
|
|
|
|
|
|
|
BuiltinVisitor(DenseHashMap<AstExprCall*, int>& result, const DenseHashMap<AstName, Global>& globals,
|
|
|
|
const DenseHashMap<AstLocal*, Variable>& variables, const CompileOptions& options)
|
|
|
|
: result(result)
|
|
|
|
, globals(globals)
|
|
|
|
, variables(variables)
|
|
|
|
, options(options)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool visit(AstExprCall* node) override
|
|
|
|
{
|
|
|
|
Builtin builtin = node->self ? Builtin() : getBuiltin(node->func, globals, variables);
|
|
|
|
if (builtin.empty())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
int bfid = getBuiltinFunctionId(builtin, options);
|
|
|
|
|
|
|
|
// getBuiltinFunctionId optimistically assumes all select() calls are builtin but actually the second argument must be a vararg
|
|
|
|
if (bfid == LBF_SELECT_VARARG && !(node->args.size == 2 && node->args.data[1]->is<AstExprVarargs>()))
|
|
|
|
bfid = -1;
|
|
|
|
|
|
|
|
if (bfid >= 0)
|
|
|
|
result[node] = bfid;
|
|
|
|
|
|
|
|
return true; // propagate to nested calls
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
void analyzeBuiltins(DenseHashMap<AstExprCall*, int>& result, const DenseHashMap<AstName, Global>& globals,
|
|
|
|
const DenseHashMap<AstLocal*, Variable>& variables, const CompileOptions& options, AstNode* root)
|
|
|
|
{
|
|
|
|
BuiltinVisitor visitor{result, globals, variables, options};
|
|
|
|
root->visit(&visitor);
|
|
|
|
}
|
|
|
|
|
2022-01-14 16:20:09 +00:00
|
|
|
} // namespace Compile
|
|
|
|
} // namespace Luau
|