mirror of
https://github.com/luau-lang/luau.git
synced 2024-12-14 06:00:39 +00:00
1100 lines
26 KiB
C++
1100 lines
26 KiB
C++
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
||
|
// This code is based on Lua 5.x implementation licensed under MIT License; see lua_LICENSE.txt for details
|
||
|
#include "lbuiltins.h"
|
||
|
|
||
|
#include "lstate.h"
|
||
|
#include "lstring.h"
|
||
|
#include "ltable.h"
|
||
|
#include "lgc.h"
|
||
|
#include "lnumutils.h"
|
||
|
#include "ldo.h"
|
||
|
|
||
|
#include <math.h>
|
||
|
|
||
|
#ifdef _MSC_VER
|
||
|
#include <intrin.h>
|
||
|
#endif
|
||
|
|
||
|
// luauF functions implement FASTCALL instruction that performs a direct execution of some builtin functions from the VM
|
||
|
// The rule of thumb is that FASTCALL functions can not call user code, yield, fail, or reallocate stack.
|
||
|
// If types of the arguments mismatch, luauF_* needs to return -1 and the execution will fall back to the usual call path
|
||
|
// If luauF_* succeeds, it needs to return *all* requested arguments, filling results with nil as appropriate.
|
||
|
// On input, nparams refers to the actual number of arguments (0+), whereas nresults contains LUA_MULTRET for arbitrary returns or 0+ for a
|
||
|
// fixed-length return Because of this, and the fact that "extra" returned values will be ignored, implementations below typically check that nresults
|
||
|
// is <= expected number, which covers the LUA_MULTRET case.
|
||
|
|
||
|
static int luauF_assert(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults == 0 && !l_isfalse(arg0))
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_abs(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
setnvalue(res, fabs(a1));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_acos(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
setnvalue(res, acos(a1));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_asin(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
setnvalue(res, asin(a1));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_atan2(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 2 && nresults <= 1 && ttisnumber(arg0) && ttisnumber(args))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
double a2 = nvalue(args);
|
||
|
setnvalue(res, atan2(a1, a2));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_atan(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
setnvalue(res, atan(a1));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
LUAU_FASTMATH_BEGIN
|
||
|
static int luauF_ceil(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
setnvalue(res, ceil(a1));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
LUAU_FASTMATH_END
|
||
|
|
||
|
static int luauF_cosh(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
setnvalue(res, cosh(a1));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_cos(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
setnvalue(res, cos(a1));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_deg(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
const double rpd = (3.14159265358979323846 / 180.0);
|
||
|
setnvalue(res, a1 / rpd);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_exp(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
setnvalue(res, exp(a1));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
LUAU_FASTMATH_BEGIN
|
||
|
static int luauF_floor(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
setnvalue(res, floor(a1));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
LUAU_FASTMATH_END
|
||
|
|
||
|
static int luauF_fmod(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 2 && nresults <= 1 && ttisnumber(arg0) && ttisnumber(args))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
double a2 = nvalue(args);
|
||
|
setnvalue(res, fmod(a1, a2));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_frexp(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 2 && ttisnumber(arg0))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
int e;
|
||
|
double f = frexp(a1, &e);
|
||
|
setnvalue(res, f);
|
||
|
setnvalue(res + 1, double(e));
|
||
|
return 2;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_ldexp(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 2 && nresults <= 1 && ttisnumber(arg0) && ttisnumber(args))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
double a2 = nvalue(args);
|
||
|
setnvalue(res, ldexp(a1, int(a2)));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_log10(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
setnvalue(res, log10(a1));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_log(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
|
||
|
if (nparams == 1)
|
||
|
{
|
||
|
setnvalue(res, log(a1));
|
||
|
return 1;
|
||
|
}
|
||
|
else if (ttisnumber(args))
|
||
|
{
|
||
|
double a2 = nvalue(args);
|
||
|
|
||
|
if (a2 == 2.0)
|
||
|
{
|
||
|
setnvalue(res, log2(a1));
|
||
|
return 1;
|
||
|
}
|
||
|
else if (a2 == 10.0)
|
||
|
{
|
||
|
setnvalue(res, log10(a1));
|
||
|
return 1;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
setnvalue(res, log(a1) / log(a2));
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_max(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double r = nvalue(arg0);
|
||
|
|
||
|
for (int i = 2; i <= nparams; ++i)
|
||
|
{
|
||
|
if (!ttisnumber(args + (i - 2)))
|
||
|
return -1;
|
||
|
|
||
|
double a = nvalue(args + (i - 2));
|
||
|
|
||
|
r = (a > r) ? a : r;
|
||
|
}
|
||
|
|
||
|
setnvalue(res, r);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_min(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double r = nvalue(arg0);
|
||
|
|
||
|
for (int i = 2; i <= nparams; ++i)
|
||
|
{
|
||
|
if (!ttisnumber(args + (i - 2)))
|
||
|
return -1;
|
||
|
|
||
|
double a = nvalue(args + (i - 2));
|
||
|
|
||
|
r = (a < r) ? a : r;
|
||
|
}
|
||
|
|
||
|
setnvalue(res, r);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_modf(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 2 && ttisnumber(arg0))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
double ip;
|
||
|
double fp = modf(a1, &ip);
|
||
|
setnvalue(res, ip);
|
||
|
setnvalue(res + 1, fp);
|
||
|
return 2;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_pow(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 2 && nresults <= 1 && ttisnumber(arg0) && ttisnumber(args))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
double a2 = nvalue(args);
|
||
|
setnvalue(res, pow(a1, a2));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_rad(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
const double rpd = (3.14159265358979323846 / 180.0);
|
||
|
setnvalue(res, a1 * rpd);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_sinh(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
setnvalue(res, sinh(a1));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_sin(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
setnvalue(res, sin(a1));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
LUAU_FASTMATH_BEGIN
|
||
|
static int luauF_sqrt(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
setnvalue(res, sqrt(a1));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
LUAU_FASTMATH_END
|
||
|
|
||
|
static int luauF_tanh(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
setnvalue(res, tanh(a1));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_tan(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
setnvalue(res, tan(a1));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_arshift(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 2 && nresults <= 1 && ttisnumber(arg0) && ttisnumber(args))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
double a2 = nvalue(args);
|
||
|
|
||
|
unsigned u;
|
||
|
luai_num2unsigned(u, a1);
|
||
|
int s = int(a2);
|
||
|
|
||
|
// note: we only specialize fast-path that doesn't require further conditionals (negative shifts and shifts greater or equal to bit width can
|
||
|
// be handled generically)
|
||
|
if (unsigned(s) < 32)
|
||
|
{
|
||
|
// note: technically right shift of negative values is UB, but this behavior is getting defined in C++20 and all compilers do the right
|
||
|
// (shift) thing.
|
||
|
uint32_t r = int32_t(u) >> s;
|
||
|
|
||
|
setnvalue(res, double(r));
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_band(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1)
|
||
|
{
|
||
|
uint32_t r = ~0u;
|
||
|
|
||
|
if (!ttisnumber(arg0))
|
||
|
return -1;
|
||
|
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
unsigned u;
|
||
|
luai_num2unsigned(u, a1);
|
||
|
|
||
|
r &= u;
|
||
|
}
|
||
|
|
||
|
for (int i = 2; i <= nparams; ++i)
|
||
|
{
|
||
|
if (!ttisnumber(args + (i - 2)))
|
||
|
return -1;
|
||
|
|
||
|
double a = nvalue(args + (i - 2));
|
||
|
unsigned u;
|
||
|
luai_num2unsigned(u, a);
|
||
|
|
||
|
r &= u;
|
||
|
}
|
||
|
|
||
|
setnvalue(res, double(r));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_bnot(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
unsigned u;
|
||
|
luai_num2unsigned(u, a1);
|
||
|
|
||
|
uint32_t r = ~u;
|
||
|
|
||
|
setnvalue(res, double(r));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_bor(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1)
|
||
|
{
|
||
|
uint32_t r = 0;
|
||
|
|
||
|
if (!ttisnumber(arg0))
|
||
|
return -1;
|
||
|
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
unsigned u;
|
||
|
luai_num2unsigned(u, a1);
|
||
|
|
||
|
r |= u;
|
||
|
}
|
||
|
|
||
|
for (int i = 2; i <= nparams; ++i)
|
||
|
{
|
||
|
if (!ttisnumber(args + (i - 2)))
|
||
|
return -1;
|
||
|
|
||
|
double a = nvalue(args + (i - 2));
|
||
|
unsigned u;
|
||
|
luai_num2unsigned(u, a);
|
||
|
|
||
|
r |= u;
|
||
|
}
|
||
|
|
||
|
setnvalue(res, double(r));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_bxor(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1)
|
||
|
{
|
||
|
uint32_t r = 0;
|
||
|
|
||
|
if (!ttisnumber(arg0))
|
||
|
return -1;
|
||
|
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
unsigned u;
|
||
|
luai_num2unsigned(u, a1);
|
||
|
|
||
|
r ^= u;
|
||
|
}
|
||
|
|
||
|
for (int i = 2; i <= nparams; ++i)
|
||
|
{
|
||
|
if (!ttisnumber(args + (i - 2)))
|
||
|
return -1;
|
||
|
|
||
|
double a = nvalue(args + (i - 2));
|
||
|
unsigned u;
|
||
|
luai_num2unsigned(u, a);
|
||
|
|
||
|
r ^= u;
|
||
|
}
|
||
|
|
||
|
setnvalue(res, double(r));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_btest(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1)
|
||
|
{
|
||
|
uint32_t r = ~0u;
|
||
|
|
||
|
if (!ttisnumber(arg0))
|
||
|
return -1;
|
||
|
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
unsigned u;
|
||
|
luai_num2unsigned(u, a1);
|
||
|
|
||
|
r &= u;
|
||
|
}
|
||
|
|
||
|
for (int i = 2; i <= nparams; ++i)
|
||
|
{
|
||
|
if (!ttisnumber(args + (i - 2)))
|
||
|
return -1;
|
||
|
|
||
|
double a = nvalue(args + (i - 2));
|
||
|
unsigned u;
|
||
|
luai_num2unsigned(u, a);
|
||
|
|
||
|
r &= u;
|
||
|
}
|
||
|
|
||
|
setbvalue(res, r != 0);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_extract(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 3 && nresults <= 1 && ttisnumber(arg0) && ttisnumber(args) && ttisnumber(args + 1))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
double a2 = nvalue(args);
|
||
|
double a3 = nvalue(args + 1);
|
||
|
|
||
|
unsigned n;
|
||
|
luai_num2unsigned(n, a1);
|
||
|
int f = int(a2);
|
||
|
int w = int(a3);
|
||
|
|
||
|
if (f >= 0 && w > 0 && f + w <= 32)
|
||
|
{
|
||
|
uint32_t m = ~(0xfffffffeu << (w - 1));
|
||
|
uint32_t r = (n >> f) & m;
|
||
|
|
||
|
setnvalue(res, double(r));
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_lrotate(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 2 && nresults <= 1 && ttisnumber(arg0) && ttisnumber(args))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
double a2 = nvalue(args);
|
||
|
|
||
|
unsigned u;
|
||
|
luai_num2unsigned(u, a1);
|
||
|
int s = int(a2);
|
||
|
|
||
|
// MSVC doesn't recognize the rotate form that is UB-safe
|
||
|
#ifdef _MSC_VER
|
||
|
uint32_t r = _rotl(u, s);
|
||
|
#else
|
||
|
uint32_t r = (u << (s & 31)) | (u >> ((32 - s) & 31));
|
||
|
#endif
|
||
|
|
||
|
setnvalue(res, double(r));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_lshift(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 2 && nresults <= 1 && ttisnumber(arg0) && ttisnumber(args))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
double a2 = nvalue(args);
|
||
|
|
||
|
unsigned u;
|
||
|
luai_num2unsigned(u, a1);
|
||
|
int s = int(a2);
|
||
|
|
||
|
// note: we only specialize fast-path that doesn't require further conditionals (negative shifts and shifts greater or equal to bit width can
|
||
|
// be handled generically)
|
||
|
if (unsigned(s) < 32)
|
||
|
{
|
||
|
uint32_t r = u << s;
|
||
|
|
||
|
setnvalue(res, double(r));
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_replace(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 4 && nresults <= 1 && ttisnumber(arg0) && ttisnumber(args) && ttisnumber(args + 1) && ttisnumber(args + 2))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
double a2 = nvalue(args);
|
||
|
double a3 = nvalue(args + 1);
|
||
|
double a4 = nvalue(args + 2);
|
||
|
|
||
|
unsigned n, v;
|
||
|
luai_num2unsigned(n, a1);
|
||
|
luai_num2unsigned(v, a2);
|
||
|
int f = int(a3);
|
||
|
int w = int(a4);
|
||
|
|
||
|
if (f >= 0 && w > 0 && f + w <= 32)
|
||
|
{
|
||
|
uint32_t m = ~(0xfffffffeu << (w - 1));
|
||
|
uint32_t r = (n & ~(m << f)) | ((v & m) << f);
|
||
|
|
||
|
setnvalue(res, double(r));
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_rrotate(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 2 && nresults <= 1 && ttisnumber(arg0) && ttisnumber(args))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
double a2 = nvalue(args);
|
||
|
|
||
|
unsigned u;
|
||
|
luai_num2unsigned(u, a1);
|
||
|
int s = int(a2);
|
||
|
|
||
|
// MSVC doesn't recognize the rotate form that is UB-safe
|
||
|
#ifdef _MSC_VER
|
||
|
uint32_t r = _rotr(u, s);
|
||
|
#else
|
||
|
uint32_t r = (u >> (s & 31)) | (u << ((32 - s) & 31));
|
||
|
#endif
|
||
|
|
||
|
setnvalue(res, double(r));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_rshift(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 2 && nresults <= 1 && ttisnumber(arg0) && ttisnumber(args))
|
||
|
{
|
||
|
double a1 = nvalue(arg0);
|
||
|
double a2 = nvalue(args);
|
||
|
|
||
|
unsigned u;
|
||
|
luai_num2unsigned(u, a1);
|
||
|
int s = int(a2);
|
||
|
|
||
|
// note: we only specialize fast-path that doesn't require further conditionals (negative shifts and shifts greater or equal to bit width can
|
||
|
// be handled generically)
|
||
|
if (unsigned(s) < 32)
|
||
|
{
|
||
|
uint32_t r = u >> s;
|
||
|
|
||
|
setnvalue(res, double(r));
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_type(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1)
|
||
|
{
|
||
|
int tt = ttype(arg0);
|
||
|
TString* ttname = L->global->ttname[tt];
|
||
|
|
||
|
setsvalue2s(L, res, ttname);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_byte(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 2 && ttisstring(arg0) && ttisnumber(args))
|
||
|
{
|
||
|
TString* ts = tsvalue(arg0);
|
||
|
int i = int(nvalue(args));
|
||
|
int j = (nparams >= 3) ? (ttisnumber(args + 1) ? int(nvalue(args + 1)) : 0) : i;
|
||
|
|
||
|
if (i >= 1 && j >= i && j <= int(ts->len))
|
||
|
{
|
||
|
int c = j - i + 1;
|
||
|
const char* s = getstr(ts);
|
||
|
|
||
|
// for vararg returns, we only support a single result
|
||
|
// this is because this frees us from concerns about stack space
|
||
|
if (c == (nresults < 0 ? 1 : nresults))
|
||
|
{
|
||
|
for (int k = 0; k < c; ++k)
|
||
|
{
|
||
|
setnvalue(res + k, uint8_t(s[i + k - 1]));
|
||
|
}
|
||
|
|
||
|
return c;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_char(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
char buffer[8];
|
||
|
|
||
|
if (nparams < int(sizeof(buffer)) && nresults <= 1)
|
||
|
{
|
||
|
|
||
|
if (nparams >= 1)
|
||
|
{
|
||
|
if (!ttisnumber(arg0))
|
||
|
return -1;
|
||
|
|
||
|
int ch = int(nvalue(arg0));
|
||
|
|
||
|
if ((unsigned char)(ch) != ch)
|
||
|
return -1;
|
||
|
|
||
|
buffer[0] = ch;
|
||
|
}
|
||
|
|
||
|
for (int i = 2; i <= nparams; ++i)
|
||
|
{
|
||
|
if (!ttisnumber(args + (i - 2)))
|
||
|
return -1;
|
||
|
|
||
|
int ch = int(nvalue(args + (i - 2)));
|
||
|
|
||
|
if ((unsigned char)(ch) != ch)
|
||
|
return -1;
|
||
|
|
||
|
buffer[i - 1] = ch;
|
||
|
}
|
||
|
|
||
|
buffer[nparams] = 0;
|
||
|
|
||
|
setsvalue2s(L, res, luaS_newlstr(L, buffer, nparams));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_len(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisstring(arg0))
|
||
|
{
|
||
|
TString* ts = tsvalue(arg0);
|
||
|
|
||
|
setnvalue(res, int(ts->len));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_typeof(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1)
|
||
|
{
|
||
|
const TString* ttname = luaT_objtypenamestr(L, arg0);
|
||
|
|
||
|
setsvalue2s(L, res, ttname);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_sub(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 3 && nresults <= 1 && ttisstring(arg0) && ttisnumber(args) && ttisnumber(args + 1))
|
||
|
{
|
||
|
TString* ts = tsvalue(arg0);
|
||
|
int i = int(nvalue(args));
|
||
|
int j = int(nvalue(args + 1));
|
||
|
|
||
|
if (i >= 1 && j >= i && unsigned(j - 1) < unsigned(ts->len))
|
||
|
{
|
||
|
setsvalue2s(L, res, luaS_newlstr(L, getstr(ts) + (i - 1), j - i + 1));
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_clamp(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 3 && nresults <= 1 && ttisnumber(arg0) && ttisnumber(args) && ttisnumber(args + 1))
|
||
|
{
|
||
|
double v = nvalue(arg0);
|
||
|
double min = nvalue(args);
|
||
|
double max = nvalue(args + 1);
|
||
|
|
||
|
if (min <= max)
|
||
|
{
|
||
|
double r = v < min ? min : v;
|
||
|
r = r > max ? max : r;
|
||
|
|
||
|
setnvalue(res, r);
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_sign(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double v = nvalue(arg0);
|
||
|
setnvalue(res, v > 0.0 ? 1.0 : v < 0.0 ? -1.0 : 0.0);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
LUAU_FASTMATH_BEGIN
|
||
|
static int luauF_round(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults <= 1 && ttisnumber(arg0))
|
||
|
{
|
||
|
double v = nvalue(arg0);
|
||
|
setnvalue(res, round(v));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
LUAU_FASTMATH_END
|
||
|
|
||
|
static int luauF_rawequal(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 2 && nresults <= 1)
|
||
|
{
|
||
|
setbvalue(res, luaO_rawequalObj(arg0, args));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_rawget(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 2 && nresults <= 1 && ttistable(arg0))
|
||
|
{
|
||
|
setobj2s(L, res, luaH_get(hvalue(arg0), args));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_rawset(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 3 && nresults <= 1 && ttistable(arg0))
|
||
|
{
|
||
|
const TValue* key = args;
|
||
|
if (ttisnil(key))
|
||
|
return -1;
|
||
|
else if (ttisnumber(key) && luai_numisnan(nvalue(key)))
|
||
|
return -1;
|
||
|
else if (ttisvector(key) && luai_vecisnan(vvalue(key)))
|
||
|
return -1;
|
||
|
|
||
|
if (hvalue(arg0)->readonly)
|
||
|
return -1;
|
||
|
|
||
|
setobj2s(L, res, arg0);
|
||
|
setobj2t(L, luaH_set(L, hvalue(arg0), args), args + 1);
|
||
|
luaC_barriert(L, hvalue(arg0), args + 1);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_tinsert(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams == 2 && nresults <= 0 && ttistable(arg0))
|
||
|
{
|
||
|
if (hvalue(arg0)->readonly)
|
||
|
return -1;
|
||
|
|
||
|
int pos = luaH_getn(hvalue(arg0)) + 1;
|
||
|
setobj2t(L, luaH_setnum(L, hvalue(arg0), pos), args);
|
||
|
luaC_barriert(L, hvalue(arg0), args);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_tunpack(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 1 && nresults < 0 && ttistable(arg0))
|
||
|
{
|
||
|
Table* t = hvalue(arg0);
|
||
|
int n = -1;
|
||
|
|
||
|
if (nparams == 1)
|
||
|
n = luaH_getn(t);
|
||
|
else if (nparams == 3 && ttisnumber(args) && ttisnumber(args + 1) && nvalue(args) == 1.0)
|
||
|
n = int(nvalue(args + 1));
|
||
|
|
||
|
if (n >= 0 && n <= t->sizearray && cast_int(L->stack_last - res) >= n)
|
||
|
{
|
||
|
TValue* array = t->array;
|
||
|
for (int i = 0; i < n; ++i)
|
||
|
setobj2s(L, res + i, array + i);
|
||
|
expandstacklimit(L, res + n);
|
||
|
return n;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int luauF_vector(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||
|
{
|
||
|
if (nparams >= 3 && nresults <= 1 && ttisnumber(arg0) && ttisnumber(args) && ttisnumber(args + 1))
|
||
|
{
|
||
|
double x = nvalue(arg0);
|
||
|
double y = nvalue(args);
|
||
|
double z = nvalue(args + 1);
|
||
|
|
||
|
setvvalue(res, float(x), float(y), float(z));
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
luau_FastFunction luauF_table[256] = {
|
||
|
NULL,
|
||
|
luauF_assert,
|
||
|
|
||
|
luauF_abs,
|
||
|
luauF_acos,
|
||
|
luauF_asin,
|
||
|
luauF_atan2,
|
||
|
luauF_atan,
|
||
|
luauF_ceil,
|
||
|
luauF_cosh,
|
||
|
luauF_cos,
|
||
|
luauF_deg,
|
||
|
luauF_exp,
|
||
|
luauF_floor,
|
||
|
luauF_fmod,
|
||
|
luauF_frexp,
|
||
|
luauF_ldexp,
|
||
|
luauF_log10,
|
||
|
luauF_log,
|
||
|
luauF_max,
|
||
|
luauF_min,
|
||
|
luauF_modf,
|
||
|
luauF_pow,
|
||
|
luauF_rad,
|
||
|
luauF_sinh,
|
||
|
luauF_sin,
|
||
|
luauF_sqrt,
|
||
|
luauF_tanh,
|
||
|
luauF_tan,
|
||
|
|
||
|
luauF_arshift,
|
||
|
luauF_band,
|
||
|
luauF_bnot,
|
||
|
luauF_bor,
|
||
|
luauF_bxor,
|
||
|
luauF_btest,
|
||
|
luauF_extract,
|
||
|
luauF_lrotate,
|
||
|
luauF_lshift,
|
||
|
luauF_replace,
|
||
|
luauF_rrotate,
|
||
|
luauF_rshift,
|
||
|
|
||
|
luauF_type,
|
||
|
|
||
|
luauF_byte,
|
||
|
luauF_char,
|
||
|
luauF_len,
|
||
|
|
||
|
luauF_typeof,
|
||
|
|
||
|
luauF_sub,
|
||
|
|
||
|
luauF_clamp,
|
||
|
luauF_sign,
|
||
|
luauF_round,
|
||
|
|
||
|
luauF_rawset,
|
||
|
luauF_rawget,
|
||
|
luauF_rawequal,
|
||
|
|
||
|
luauF_tinsert,
|
||
|
luauF_tunpack,
|
||
|
|
||
|
luauF_vector,
|
||
|
};
|