// 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" LUAU_FASTFLAGVARIABLE(LuauCompileSelectBuiltin, false) namespace Luau { namespace Compile { Builtin getBuiltin(AstExpr* node, const DenseHashMap& globals, const DenseHashMap& variables) { if (AstExprLocal* expr = node->as()) { 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()) { if (AstExprGlobal* object = expr->expr->as()) { return getGlobalState(globals, object->name) == Global::Default ? Builtin{object->name, expr->index} : Builtin(); } else { return Builtin(); } } else if (AstExprGlobal* expr = node->as()) { return getGlobalState(globals, expr->name) == Global::Default ? Builtin{AstName(), expr->name} : Builtin(); } else { return Builtin(); } } int getBuiltinFunctionId(const Builtin& builtin, const CompileOptions& options) { if (builtin.empty()) return -1; 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; if (builtin.isGlobal("unpack")) return LBF_TABLE_UNPACK; if (FFlag::LuauCompileSelectBuiltin && builtin.isGlobal("select")) return LBF_SELECT_VARARG; 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; } } // namespace Compile } // namespace Luau