mirror of
https://github.com/luau-lang/luau.git
synced 2024-12-13 21:40:43 +00:00
76f67e0733
Type checker/autocomplete: * `Luau::autocomplete` no longer performs typechecking internally, make sure to run `Frontend::check` before performing autocomplete requests * Autocomplete string suggestions without "" are now only suggested inside the "" * Autocomplete suggestions now include `function (anonymous autofilled)` key with a full suggestion for the function expression (with arguments included) stored in `AutocompleteEntry::insertText` * `AutocompleteEntry::indexedWithSelf` is provided for function call suggestions made with `:` * Cyclic modules now see each other type exports as `any` to prevent memory use-after-free (similar to module return type) Runtime: * Updated inline/loop unroll cost model to better handle assignments (Fixes https://github.com/Roblox/luau/issues/978) * `math.noise` speed was improved by ~30% * `table.concat` speed was improved by ~5-7% * `tonumber` and `tostring` now have fastcall paths that execute ~1.5x and ~2.5x faster respectively (fixes #777) * Fixed crash in `luaL_typename` when index refers to a non-existing value * Fixed potential out of memory scenario when using `string.sub` or `string.char` in a loop * Fixed behavior of some fastcall builtins when called without arguments under -O2 to match original functions * Support for native code execution in VM is now enabled by default (note: native code still has to be generated explicitly) * `Codegen::compile` now accepts `CodeGen_OnlyNativeModules` flag. When set, only modules that have a `--!native` hot-comment at the top will be compiled to native code In our new typechecker: * Generic type packs are no longer considered to be variadic during unification * Timeout and cancellation now works in new solver * Fixed false positive errors around 'table' and 'function' type refinements * Table literals now use covariant unification rules. This is sound since literal has no type specified and has no aliases * Fixed issues with blocked types escaping the constraint solver * Fixed more places where error messages that should've been suppressed were still reported * Fixed errors when iterating over a top table type In our native code generation (jit): * 'DebugLuauAbortingChecks' flag is now supported on A64 * LOP_NEWCLOSURE has been translated to IR
471 lines
11 KiB
C
471 lines
11 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
|
|
#pragma once
|
|
|
|
#include "lua.h"
|
|
#include "lcommon.h"
|
|
|
|
/*
|
|
** Union of all collectible objects
|
|
*/
|
|
typedef union GCObject GCObject;
|
|
|
|
/*
|
|
** Common Header for all collectible objects (in macro form, to be included in other objects)
|
|
*/
|
|
// clang-format off
|
|
#define CommonHeader \
|
|
uint8_t tt; uint8_t marked; uint8_t memcat
|
|
// clang-format on
|
|
|
|
/*
|
|
** Common header in struct form
|
|
*/
|
|
typedef struct GCheader
|
|
{
|
|
CommonHeader;
|
|
} GCheader;
|
|
|
|
/*
|
|
** Union of all Lua values
|
|
*/
|
|
typedef union
|
|
{
|
|
GCObject* gc;
|
|
void* p;
|
|
double n;
|
|
int b;
|
|
float v[2]; // v[0], v[1] live here; v[2] lives in TValue::extra
|
|
} Value;
|
|
|
|
/*
|
|
** Tagged Values
|
|
*/
|
|
|
|
typedef struct lua_TValue
|
|
{
|
|
Value value;
|
|
int extra[LUA_EXTRA_SIZE];
|
|
int tt;
|
|
} TValue;
|
|
|
|
// Macros to test type
|
|
#define ttisnil(o) (ttype(o) == LUA_TNIL)
|
|
#define ttisnumber(o) (ttype(o) == LUA_TNUMBER)
|
|
#define ttisstring(o) (ttype(o) == LUA_TSTRING)
|
|
#define ttistable(o) (ttype(o) == LUA_TTABLE)
|
|
#define ttisfunction(o) (ttype(o) == LUA_TFUNCTION)
|
|
#define ttisboolean(o) (ttype(o) == LUA_TBOOLEAN)
|
|
#define ttisuserdata(o) (ttype(o) == LUA_TUSERDATA)
|
|
#define ttisthread(o) (ttype(o) == LUA_TTHREAD)
|
|
#define ttislightuserdata(o) (ttype(o) == LUA_TLIGHTUSERDATA)
|
|
#define ttisvector(o) (ttype(o) == LUA_TVECTOR)
|
|
#define ttisupval(o) (ttype(o) == LUA_TUPVAL)
|
|
|
|
// Macros to access values
|
|
#define ttype(o) ((o)->tt)
|
|
#define gcvalue(o) check_exp(iscollectable(o), (o)->value.gc)
|
|
#define pvalue(o) check_exp(ttislightuserdata(o), (o)->value.p)
|
|
#define nvalue(o) check_exp(ttisnumber(o), (o)->value.n)
|
|
#define vvalue(o) check_exp(ttisvector(o), (o)->value.v)
|
|
#define tsvalue(o) check_exp(ttisstring(o), &(o)->value.gc->ts)
|
|
#define uvalue(o) check_exp(ttisuserdata(o), &(o)->value.gc->u)
|
|
#define clvalue(o) check_exp(ttisfunction(o), &(o)->value.gc->cl)
|
|
#define hvalue(o) check_exp(ttistable(o), &(o)->value.gc->h)
|
|
#define bvalue(o) check_exp(ttisboolean(o), (o)->value.b)
|
|
#define thvalue(o) check_exp(ttisthread(o), &(o)->value.gc->th)
|
|
#define upvalue(o) check_exp(ttisupval(o), &(o)->value.gc->uv)
|
|
|
|
#define l_isfalse(o) (ttisnil(o) || (ttisboolean(o) && bvalue(o) == 0))
|
|
|
|
/*
|
|
** for internal debug only
|
|
*/
|
|
#define checkconsistency(obj) LUAU_ASSERT(!iscollectable(obj) || (ttype(obj) == (obj)->value.gc->gch.tt))
|
|
|
|
#define checkliveness(g, obj) LUAU_ASSERT(!iscollectable(obj) || ((ttype(obj) == (obj)->value.gc->gch.tt) && !isdead(g, (obj)->value.gc)))
|
|
|
|
// Macros to set values
|
|
#define setnilvalue(obj) ((obj)->tt = LUA_TNIL)
|
|
|
|
#define setnvalue(obj, x) \
|
|
{ \
|
|
TValue* i_o = (obj); \
|
|
i_o->value.n = (x); \
|
|
i_o->tt = LUA_TNUMBER; \
|
|
}
|
|
|
|
#if LUA_VECTOR_SIZE == 4
|
|
#define setvvalue(obj, x, y, z, w) \
|
|
{ \
|
|
TValue* i_o = (obj); \
|
|
float* i_v = i_o->value.v; \
|
|
i_v[0] = (x); \
|
|
i_v[1] = (y); \
|
|
i_v[2] = (z); \
|
|
i_v[3] = (w); \
|
|
i_o->tt = LUA_TVECTOR; \
|
|
}
|
|
#else
|
|
#define setvvalue(obj, x, y, z, w) \
|
|
{ \
|
|
TValue* i_o = (obj); \
|
|
float* i_v = i_o->value.v; \
|
|
i_v[0] = (x); \
|
|
i_v[1] = (y); \
|
|
i_v[2] = (z); \
|
|
i_o->tt = LUA_TVECTOR; \
|
|
}
|
|
#endif
|
|
|
|
#define setpvalue(obj, x) \
|
|
{ \
|
|
TValue* i_o = (obj); \
|
|
i_o->value.p = (x); \
|
|
i_o->tt = LUA_TLIGHTUSERDATA; \
|
|
}
|
|
|
|
#define setbvalue(obj, x) \
|
|
{ \
|
|
TValue* i_o = (obj); \
|
|
i_o->value.b = (x); \
|
|
i_o->tt = LUA_TBOOLEAN; \
|
|
}
|
|
|
|
#define setsvalue(L, obj, x) \
|
|
{ \
|
|
TValue* i_o = (obj); \
|
|
i_o->value.gc = cast_to(GCObject*, (x)); \
|
|
i_o->tt = LUA_TSTRING; \
|
|
checkliveness(L->global, i_o); \
|
|
}
|
|
|
|
#define setuvalue(L, obj, x) \
|
|
{ \
|
|
TValue* i_o = (obj); \
|
|
i_o->value.gc = cast_to(GCObject*, (x)); \
|
|
i_o->tt = LUA_TUSERDATA; \
|
|
checkliveness(L->global, i_o); \
|
|
}
|
|
|
|
#define setthvalue(L, obj, x) \
|
|
{ \
|
|
TValue* i_o = (obj); \
|
|
i_o->value.gc = cast_to(GCObject*, (x)); \
|
|
i_o->tt = LUA_TTHREAD; \
|
|
checkliveness(L->global, i_o); \
|
|
}
|
|
|
|
#define setclvalue(L, obj, x) \
|
|
{ \
|
|
TValue* i_o = (obj); \
|
|
i_o->value.gc = cast_to(GCObject*, (x)); \
|
|
i_o->tt = LUA_TFUNCTION; \
|
|
checkliveness(L->global, i_o); \
|
|
}
|
|
|
|
#define sethvalue(L, obj, x) \
|
|
{ \
|
|
TValue* i_o = (obj); \
|
|
i_o->value.gc = cast_to(GCObject*, (x)); \
|
|
i_o->tt = LUA_TTABLE; \
|
|
checkliveness(L->global, i_o); \
|
|
}
|
|
|
|
#define setptvalue(L, obj, x) \
|
|
{ \
|
|
TValue* i_o = (obj); \
|
|
i_o->value.gc = cast_to(GCObject*, (x)); \
|
|
i_o->tt = LUA_TPROTO; \
|
|
checkliveness(L->global, i_o); \
|
|
}
|
|
|
|
#define setupvalue(L, obj, x) \
|
|
{ \
|
|
TValue* i_o = (obj); \
|
|
i_o->value.gc = cast_to(GCObject*, (x)); \
|
|
i_o->tt = LUA_TUPVAL; \
|
|
checkliveness(L->global, i_o); \
|
|
}
|
|
|
|
#define setobj(L, obj1, obj2) \
|
|
{ \
|
|
const TValue* o2 = (obj2); \
|
|
TValue* o1 = (obj1); \
|
|
*o1 = *o2; \
|
|
checkliveness(L->global, o1); \
|
|
}
|
|
|
|
/*
|
|
** different types of sets, according to destination
|
|
*/
|
|
|
|
// to stack
|
|
#define setobj2s setobj
|
|
// from table to same table (no barrier)
|
|
#define setobjt2t setobj
|
|
// to table (needs barrier)
|
|
#define setobj2t setobj
|
|
// to new object (no barrier)
|
|
#define setobj2n setobj
|
|
|
|
#define setttype(obj, tt) (ttype(obj) = (tt))
|
|
|
|
#define iscollectable(o) (ttype(o) >= LUA_TSTRING)
|
|
|
|
typedef TValue* StkId; // index to stack elements
|
|
|
|
/*
|
|
** String headers for string table
|
|
*/
|
|
typedef struct TString
|
|
{
|
|
CommonHeader;
|
|
// 1 byte padding
|
|
|
|
int16_t atom;
|
|
// 2 byte padding
|
|
|
|
TString* next; // next string in the hash table bucket
|
|
|
|
unsigned int hash;
|
|
unsigned int len;
|
|
|
|
char data[1]; // string data is allocated right after the header
|
|
} TString;
|
|
|
|
#define getstr(ts) (ts)->data
|
|
#define svalue(o) getstr(tsvalue(o))
|
|
|
|
typedef struct Udata
|
|
{
|
|
CommonHeader;
|
|
|
|
uint8_t tag;
|
|
|
|
int len;
|
|
|
|
struct Table* metatable;
|
|
|
|
union
|
|
{
|
|
char data[1]; // userdata is allocated right after the header
|
|
L_Umaxalign dummy; // ensures maximum alignment for data
|
|
};
|
|
} Udata;
|
|
|
|
/*
|
|
** Function Prototypes
|
|
*/
|
|
// clang-format off
|
|
typedef struct Proto
|
|
{
|
|
CommonHeader;
|
|
|
|
|
|
uint8_t nups; // number of upvalues
|
|
uint8_t numparams;
|
|
uint8_t is_vararg;
|
|
uint8_t maxstacksize;
|
|
uint8_t flags;
|
|
|
|
|
|
TValue* k; // constants used by the function
|
|
Instruction* code; // function bytecode
|
|
struct Proto** p; // functions defined inside the function
|
|
const Instruction* codeentry;
|
|
|
|
void* execdata;
|
|
uintptr_t exectarget;
|
|
|
|
|
|
uint8_t* lineinfo; // for each instruction, line number as a delta from baseline
|
|
int* abslineinfo; // baseline line info, one entry for each 1<<linegaplog2 instructions; allocated after lineinfo
|
|
struct LocVar* locvars; // information about local variables
|
|
TString** upvalues; // upvalue names
|
|
TString* source;
|
|
|
|
TString* debugname;
|
|
uint8_t* debuginsn; // a copy of code[] array with just opcodes
|
|
|
|
uint8_t* typeinfo;
|
|
|
|
void* userdata;
|
|
|
|
GCObject* gclist;
|
|
|
|
|
|
int sizecode;
|
|
int sizep;
|
|
int sizelocvars;
|
|
int sizeupvalues;
|
|
int sizek;
|
|
int sizelineinfo;
|
|
int linegaplog2;
|
|
int linedefined;
|
|
int bytecodeid;
|
|
} Proto;
|
|
// clang-format on
|
|
|
|
typedef struct LocVar
|
|
{
|
|
TString* varname;
|
|
int startpc; // first point where variable is active
|
|
int endpc; // first point where variable is dead
|
|
uint8_t reg; // register slot, relative to base, where variable is stored
|
|
} LocVar;
|
|
|
|
/*
|
|
** Upvalues
|
|
*/
|
|
|
|
typedef struct UpVal
|
|
{
|
|
CommonHeader;
|
|
uint8_t markedopen; // set if reachable from an alive thread (only valid during atomic)
|
|
|
|
// 4 byte padding (x64)
|
|
|
|
TValue* v; // points to stack or to its own value
|
|
union
|
|
{
|
|
TValue value; // the value (when closed)
|
|
struct
|
|
{
|
|
// global double linked list (when open)
|
|
struct UpVal* prev;
|
|
struct UpVal* next;
|
|
|
|
// thread linked list (when open)
|
|
struct UpVal* threadnext;
|
|
} open;
|
|
} u;
|
|
} UpVal;
|
|
|
|
#define upisopen(up) ((up)->v != &(up)->u.value)
|
|
|
|
/*
|
|
** Closures
|
|
*/
|
|
|
|
typedef struct Closure
|
|
{
|
|
CommonHeader;
|
|
|
|
uint8_t isC;
|
|
uint8_t nupvalues;
|
|
uint8_t stacksize;
|
|
uint8_t preload;
|
|
|
|
GCObject* gclist;
|
|
struct Table* env;
|
|
|
|
union
|
|
{
|
|
struct
|
|
{
|
|
lua_CFunction f;
|
|
lua_Continuation cont;
|
|
const char* debugname;
|
|
TValue upvals[1];
|
|
} c;
|
|
|
|
struct
|
|
{
|
|
struct Proto* p;
|
|
TValue uprefs[1];
|
|
} l;
|
|
};
|
|
} Closure;
|
|
|
|
#define iscfunction(o) (ttype(o) == LUA_TFUNCTION && clvalue(o)->isC)
|
|
#define isLfunction(o) (ttype(o) == LUA_TFUNCTION && !clvalue(o)->isC)
|
|
|
|
/*
|
|
** Tables
|
|
*/
|
|
|
|
typedef struct TKey
|
|
{
|
|
::Value value;
|
|
int extra[LUA_EXTRA_SIZE];
|
|
unsigned tt : 4;
|
|
int next : 28; // for chaining
|
|
} TKey;
|
|
|
|
typedef struct LuaNode
|
|
{
|
|
TValue val;
|
|
TKey key;
|
|
} LuaNode;
|
|
|
|
// copy a value into a key
|
|
#define setnodekey(L, node, obj) \
|
|
{ \
|
|
LuaNode* n_ = (node); \
|
|
const TValue* i_o = (obj); \
|
|
n_->key.value = i_o->value; \
|
|
memcpy(n_->key.extra, i_o->extra, sizeof(n_->key.extra)); \
|
|
n_->key.tt = i_o->tt; \
|
|
checkliveness(L->global, i_o); \
|
|
}
|
|
|
|
// copy a value from a key
|
|
#define getnodekey(L, obj, node) \
|
|
{ \
|
|
TValue* i_o = (obj); \
|
|
const LuaNode* n_ = (node); \
|
|
i_o->value = n_->key.value; \
|
|
memcpy(i_o->extra, n_->key.extra, sizeof(i_o->extra)); \
|
|
i_o->tt = n_->key.tt; \
|
|
checkliveness(L->global, i_o); \
|
|
}
|
|
|
|
// clang-format off
|
|
typedef struct Table
|
|
{
|
|
CommonHeader;
|
|
|
|
|
|
uint8_t tmcache; // 1<<p means tagmethod(p) is not present
|
|
uint8_t readonly; // sandboxing feature to prohibit writes to table
|
|
uint8_t safeenv; // environment doesn't share globals with other scripts
|
|
uint8_t lsizenode; // log2 of size of `node' array
|
|
uint8_t nodemask8; // (1<<lsizenode)-1, truncated to 8 bits
|
|
|
|
int sizearray; // size of `array' array
|
|
union
|
|
{
|
|
int lastfree; // any free position is before this position
|
|
int aboundary; // negated 'boundary' of `array' array; iff aboundary < 0
|
|
};
|
|
|
|
|
|
struct Table* metatable;
|
|
TValue* array; // array part
|
|
LuaNode* node;
|
|
GCObject* gclist;
|
|
} Table;
|
|
// clang-format on
|
|
|
|
/*
|
|
** `module' operation for hashing (size is always a power of 2)
|
|
*/
|
|
#define lmod(s, size) (check_exp((size & (size - 1)) == 0, (cast_to(int, (s) & ((size)-1)))))
|
|
|
|
#define twoto(x) ((int)(1 << (x)))
|
|
#define sizenode(t) (twoto((t)->lsizenode))
|
|
|
|
#define luaO_nilobject (&luaO_nilobject_)
|
|
|
|
LUAI_DATA const TValue luaO_nilobject_;
|
|
|
|
#define ceillog2(x) (luaO_log2((x)-1) + 1)
|
|
|
|
LUAI_FUNC int luaO_log2(unsigned int x);
|
|
LUAI_FUNC int luaO_rawequalObj(const TValue* t1, const TValue* t2);
|
|
LUAI_FUNC int luaO_rawequalKey(const TKey* t1, const TValue* t2);
|
|
LUAI_FUNC int luaO_str2d(const char* s, double* result);
|
|
LUAI_FUNC const char* luaO_pushvfstring(lua_State* L, const char* fmt, va_list argp);
|
|
LUAI_FUNC const char* luaO_pushfstring(lua_State* L, const char* fmt, ...);
|
|
LUAI_FUNC const char* luaO_chunkid(char* buf, size_t buflen, const char* source, size_t srclen);
|