Working version of storing {k=v} style for instance tables and removing

__indices.
This commit is contained in:
Bruce Hill 2018-06-05 15:44:00 -07:00
parent 898f92d20a
commit 2320f87324

View File

@ -19,7 +19,6 @@
* hash buckets: weak-keyed map from instance userdata -> table of values * hash buckets: weak-keyed map from instance userdata -> table of values
* __buckets: weak-keyed map from instance userdata -> hash bucket (used to manage hash bucket lifetimes) * __buckets: weak-keyed map from instance userdata -> hash bucket (used to manage hash bucket lifetimes)
* __fields: list of named fields * __fields: list of named fields
* __indices: map from field names to the index in the instance table where the value is stored
* metamethods, methods, class variables, etc. * metamethods, methods, class variables, etc.
*/ */
@ -50,11 +49,11 @@ static int WEAK_VALUE_METATABLE;
// constructed classes to have the same metatable: // constructed classes to have the same metatable:
// {__new=Lcreate_instance, __tostring=function(cls) return cls.name or 'immutable(...)' end} // {__new=Lcreate_instance, __tostring=function(cls) return cls.name or 'immutable(...)' end}
static int SHARED_CLASS_METATABLE; static int SHARED_CLASS_METATABLE;
static int SUPER_METHODS; static int SUPER_METHODS, NIL_SENTINEL;
typedef struct { typedef struct {
int hash; int hash;
size_t len; size_t array_len;
} immutable_info_t; } immutable_info_t;
#define IMMUTABLE_GETVALUE(i) if (from_table) {\ #define IMMUTABLE_GETVALUE(i) if (from_table) {\
@ -68,8 +67,12 @@ typedef struct {
if (i <= num_args) {\ if (i <= num_args) {\
lua_pushvalue(L, 1+i);\ lua_pushvalue(L, 1+i);\
} else {\ } else {\
lua_pushnil(L);\ lua_pushlightuserdata(L, &NIL_SENTINEL);\
}\ }\
}\
if (lua_isnil(L, -1)) {\
lua_pop(L, 1);\
lua_pushlightuserdata(L, &NIL_SENTINEL);\
} }
static inline int _create_instance(lua_State *L, int from_table) static inline int _create_instance(lua_State *L, int from_table)
@ -79,6 +82,7 @@ static inline int _create_instance(lua_State *L, int from_table)
int fields_index = lua_gettop(L); int fields_index = lua_gettop(L);
size_t num_fields = lua_objlen(L, -1); size_t num_fields = lua_objlen(L, -1);
size_t num_values = from_table ? (num_fields + lua_objlen(L, 2)) : (num_args >= num_fields ? num_args : num_fields); size_t num_values = from_table ? (num_fields + lua_objlen(L, 2)) : (num_args >= num_fields ? num_args : num_fields);
size_t array_len = num_values - num_fields;
// Compute the hash and populate the values table: // Compute the hash and populate the values table:
// Stack: [fields] // Stack: [fields]
@ -107,11 +111,15 @@ static inline int _create_instance(lua_State *L, int from_table)
// Arbitrarily chosen values // Arbitrarily chosen values
item_hash = lua_toboolean(L, -1)? 0x82684f71 : 0x88d66f2a; item_hash = lua_toboolean(L, -1)? 0x82684f71 : 0x88d66f2a;
break; break;
case LUA_TLIGHTUSERDATA:
if (lua_touserdata(L, -1) == &NIL_SENTINEL) {
item_hash = (1000003 * type) ^ (lua_Integer)lua_topointer(L, -1);
break;
}
case LUA_TTABLE: case LUA_TTABLE:
case LUA_TFUNCTION: case LUA_TFUNCTION:
case LUA_TUSERDATA: case LUA_TUSERDATA:
case LUA_TTHREAD: case LUA_TTHREAD:
case LUA_TLIGHTUSERDATA:
item_hash = (1000003 * type) ^ (lua_Integer)lua_topointer(L, -1); item_hash = (1000003 * type) ^ (lua_Integer)lua_topointer(L, -1);
break; break;
case LUA_TSTRING: case LUA_TSTRING:
@ -163,12 +171,14 @@ static inline int _create_instance(lua_State *L, int from_table)
// Stack: [fields, __instances, bucket, hash_collider_inst, collider_table] // Stack: [fields, __instances, bucket, hash_collider_inst, collider_table]
// Shallow equality check: // Shallow equality check:
immutable_info_t *collider_info = (immutable_info_t*)lua_touserdata(L, -2); immutable_info_t *collider_info = (immutable_info_t*)lua_touserdata(L, -2);
if (collider_info->len != num_values) { if (collider_info->array_len != array_len) {
lua_pop(L, 1); lua_pop(L, 1);
goto next_bucket_item; goto next_bucket_item;
} }
for (size_t i = 1; i <= num_values; i++) { // Stack: [fields, __instances, bucket, hash_collider_inst, collider_table]
lua_rawgeti(L, -1, i); for (size_t i = 1; i <= num_fields; i++) {
lua_rawgeti(L, fields_index, i);
lua_gettable(L, -2);
IMMUTABLE_GETVALUE(i); IMMUTABLE_GETVALUE(i);
// Stack: [fields, __instances, bucket, hash_collider_inst, collider_table, collider_val, inst_val] // Stack: [fields, __instances, bucket, hash_collider_inst, collider_table, collider_val, inst_val]
if (! lua_rawequal(L, -1, -2)) { // If the i'th entry doesn't match the i'th arg if (! lua_rawequal(L, -1, -2)) { // If the i'th entry doesn't match the i'th arg
@ -178,6 +188,17 @@ static inline int _create_instance(lua_State *L, int from_table)
lua_pop(L, 2); lua_pop(L, 2);
} }
} }
for (size_t i = 1; i <= array_len; i++) {
lua_rawgeti(L, -1, i);
IMMUTABLE_GETVALUE(i+num_fields);
// Stack: [fields, __instances, bucket, hash_collider_inst, collider_table, collider_val, inst_val]
if (! lua_rawequal(L, -1, -2)) { // If the i'th entry doesn't match the i'th arg
lua_pop(L, 3);
goto next_bucket_item;
} else {
lua_pop(L, 2);
}
}
// bucket item matches // bucket item matches
// Stack: [fields, __instances, bucket, hash_collider_inst, collider_table] // Stack: [fields, __instances, bucket, hash_collider_inst, collider_table]
lua_pop(L, 1); lua_pop(L, 1);
@ -194,7 +215,7 @@ static inline int _create_instance(lua_State *L, int from_table)
// Stack: [fields, __instances, bucket, inst_userdata] // Stack: [fields, __instances, bucket, inst_userdata]
int userdata_index = lua_gettop(L); int userdata_index = lua_gettop(L);
userdata->hash = hash; userdata->hash = hash;
userdata->len = num_values; userdata->array_len = array_len;
lua_pushvalue(L, 1); lua_pushvalue(L, 1);
// Stack: [fields, __instances, bucket, inst_userdata, cls] // Stack: [fields, __instances, bucket, inst_userdata, cls]
@ -214,9 +235,14 @@ static inline int _create_instance(lua_State *L, int from_table)
lua_pushvalue(L, userdata_index); lua_pushvalue(L, userdata_index);
// Stack: [fields, __instances, bucket, inst_userdata, inst_userdata] // Stack: [fields, __instances, bucket, inst_userdata, inst_userdata]
lua_createtable(L, num_values, 0); lua_createtable(L, num_values-num_fields, num_fields);
for (size_t i=1; i <= num_values; i++) { for (size_t i=1; i <= num_fields; i++) {
lua_rawgeti(L, fields_index, i);
IMMUTABLE_GETVALUE(i); IMMUTABLE_GETVALUE(i);
lua_settable(L, -3);
}
for (size_t i=1; i <= array_len; i++) {
IMMUTABLE_GETVALUE(i+num_fields);
lua_rawseti(L, -2, i); lua_rawseti(L, -2, i);
} }
// Stack: [fields, __instances, bucket, inst_userdata, inst_userdata, inst_table] // Stack: [fields, __instances, bucket, inst_userdata, inst_userdata, inst_table]
@ -263,49 +289,24 @@ static int Llen(lua_State *L)
luaL_error(L, "invalid type"); luaL_error(L, "invalid type");
} }
immutable_info_t *info = (immutable_info_t *)lua_touserdata(L, 1); immutable_info_t *info = (immutable_info_t *)lua_touserdata(L, 1);
lua_getfield(L, -1, "__fields"); lua_pushinteger(L, info->array_len);
size_t len = info->len - lua_objlen(L, -1);
lua_pushinteger(L, len);
return 1; return 1;
} }
static int Linst_index(lua_State *L) static int Lindex(lua_State *L)
{ {
// Return inst[key], <whether inst has key> // Return inst[key], <whether inst has key>
luaL_checktype(L, 1, LUA_TUSERDATA); luaL_checktype(L, 1, LUA_TUSERDATA);
if (! lua_getmetatable(L, 1)) { if (! lua_getmetatable(L, 1)) {
luaL_error(L, "invalid type"); luaL_error(L, "invalid type");
} }
int class_index = lua_gettop(L);
// Stack: [mt] // Stack: [mt]
immutable_info_t *info = (immutable_info_t*)lua_touserdata(L, 1); immutable_info_t *info = (immutable_info_t*)lua_touserdata(L, 1);
if (! info) { if (! info) {
luaL_error(L, "invalid type"); luaL_error(L, "invalid type");
} }
lua_getfield(L, -1, "__fields");
size_t num_fields = lua_objlen(L, -1);
lua_pop(L, 1);
// Stack: [mt]
lua_getfield(L, -1, "__indices");
// Stack: [mt, indices]
lua_pushvalue(L, 2);
lua_gettable(L, -2);
// Stack: [mt, indices, index]
int index;
if (lua_isnil(L, -1)) {
if (! lua_isinteger(L, 2)) {
return 0;
}
index = lua_tointeger(L, 2) + num_fields;
} else {
index = lua_tointeger(L, -1);
}
lua_pop(L, 2);
// Stack: [mt]
if (! (0 < index && index <= (int)info->len)) {
return 0;
}
// Stack: [mt] // Stack: [mt]
lua_getfield(L, -1, "__instances"); lua_getfield(L, -1, "__instances");
// Stack: [mt, buckets] // Stack: [mt, buckets]
@ -321,28 +322,20 @@ static int Linst_index(lua_State *L)
if (lua_isnil(L, -1)) { if (lua_isnil(L, -1)) {
luaL_error(L, "Failed to find instance"); luaL_error(L, "Failed to find instance");
} }
lua_rawgeti(L, -1, index);
lua_pushboolean(L, 1);
return 2;
}
static int Lindex(lua_State *L)
{
int ret = Linst_index(L);
if (ret > 0) {
return ret;
}
// Fall back to class:
lua_getmetatable(L, 1);
// Stack: [..., mt]
lua_pushvalue(L, 2); lua_pushvalue(L, 2);
// Stack: [..., mt, key]
lua_gettable(L, -2); lua_gettable(L, -2);
// Stack: [..., mt, value] if (lua_isnil(L, -1)) {
if (! lua_isnil(L, -1)) { // Fall back to class:
lua_pushvalue(L, 2);
lua_gettable(L, class_index);
return 1; return 1;
} }
return 0; if (lua_islightuserdata(L, -1) && lua_touserdata(L, -1) == &NIL_SENTINEL) {
lua_pop(L, 1);
lua_pushnil(L);
}
lua_pushboolean(L, 1);
return 2;
} }
static int Ltostring(lua_State *L) static int Ltostring(lua_State *L)
@ -382,10 +375,13 @@ static int Ltostring(lua_State *L)
} }
int inst_table_index = lua_gettop(L); int inst_table_index = lua_gettop(L);
// Stack: [mt, buckets, bucket, inst_table] // Stack: [mt, buckets, bucket, inst_table]
lua_getfield(L, cls_index, "__fields");
int fields_index = lua_gettop(L);
lua_getglobal(L, "tostring"); lua_getglobal(L, "tostring");
int tostring_index = lua_gettop(L); int tostring_index = lua_gettop(L);
// Stack: [mt, buckets, bucket, inst_table, tostring] // Stack: [mt, buckets, bucket, inst_table, __fields, tostring]
int n = info->len, i = 1; int num_fields = lua_objlen(L, fields_index);
int n = num_fields + info->array_len, i = 1;
if (i <= n) { if (i <= n) {
goto first_list_item; goto first_list_item;
} }
@ -395,7 +391,16 @@ static int Ltostring(lua_State *L)
first_list_item: first_list_item:
lua_pushvalue(L, tostring_index); lua_pushvalue(L, tostring_index);
// Stack: [mt, buckets, bucket, inst_table, tostring, ???, tostring] // Stack: [mt, buckets, bucket, inst_table, tostring, ???, tostring]
lua_rawgeti(L, inst_table_index, i); if (i <= num_fields) {
lua_rawgeti(L, fields_index, i);
lua_gettable(L, inst_table_index);
} else {
lua_rawgeti(L, inst_table_index, i - num_fields);
}
if (lua_islightuserdata(L, -1) && lua_touserdata(L, -1) == &NIL_SENTINEL) {
lua_pop(L, 1);
lua_pushnil(L);
}
// Stack: [mt, buckets, bucket, inst_table, tostring, ???, tostring, value] // Stack: [mt, buckets, bucket, inst_table, tostring, ???, tostring, value]
int quotes = lua_type(L, -1) == LUA_TSTRING; int quotes = lua_type(L, -1) == LUA_TSTRING;
lua_call(L, 1, 1); lua_call(L, 1, 1);
@ -420,7 +425,6 @@ static int Lnexti(lua_State *L)
if (! lua_getmetatable(L, 1)) { if (! lua_getmetatable(L, 1)) {
luaL_error(L, "invalid type"); luaL_error(L, "invalid type");
} }
int cls_index = lua_gettop(L);
// Stack: [mt] // Stack: [mt]
lua_getfield(L, -1, "__instances"); lua_getfield(L, -1, "__instances");
// Stack: [mt, buckets] // Stack: [mt, buckets]
@ -428,11 +432,8 @@ static int Lnexti(lua_State *L)
if (! info) { if (! info) {
luaL_error(L, "invalid type"); luaL_error(L, "invalid type");
} }
int i = lua_isnil(L, 2) ? 1 : lua_tointeger(L, 2)+1; int i = lua_tointeger(L, 2)+1;
lua_getfield(L, cls_index, "__fields"); if (i > (int)info->array_len) {
int num_fields = lua_objlen(L, -1);
lua_pop(L, 1);
if (i + num_fields > (int)info->len) {
return 0; return 0;
} }
lua_rawgeti(L, -1, info->hash); lua_rawgeti(L, -1, info->hash);
@ -446,8 +447,12 @@ static int Lnexti(lua_State *L)
// Stack: [mt, buckets, bucket, inst_table] // Stack: [mt, buckets, bucket, inst_table]
lua_pushinteger(L, i); lua_pushinteger(L, i);
// Stack: [mt, buckets, bucket, inst_table, i] // Stack: [mt, buckets, bucket, inst_table, i]
lua_rawgeti(L, -2, i + num_fields); lua_rawgeti(L, -2, i);
// Stack: [mt, buckets, bucket, inst_table, i, table[i]] // Stack: [mt, buckets, bucket, inst_table, i, table[i]]
if (lua_islightuserdata(L, -1) && lua_touserdata(L, -1) == &NIL_SENTINEL) {
lua_pop(L, 1);
lua_pushnil(L);
}
return 2; return 2;
} }
@ -457,8 +462,8 @@ static int Lipairs(lua_State *L)
// Stack: [Lnexti] // Stack: [Lnexti]
lua_pushvalue(L, 1); lua_pushvalue(L, 1);
// Stack: [Lnexti, inst_udata] // Stack: [Lnexti, inst_udata]
lua_pushnil(L); lua_pushinteger(L, 0);
// Stack: [Lnexti, inst_udata, nil] // Stack: [Lnexti, inst_udata, 0]
return 3; return 3;
} }
@ -467,57 +472,31 @@ static int Lnext(lua_State *L)
if (! lua_getmetatable(L, 1)) { if (! lua_getmetatable(L, 1)) {
luaL_error(L, "invalid type"); luaL_error(L, "invalid type");
} }
int cls_index = lua_gettop(L);
// Stack: [mt] // Stack: [mt]
lua_getfield(L, cls_index, "__fields"); lua_getfield(L, -1, "__instances");
int fields_index = lua_gettop(L); // Stack: [mt, buckets]
lua_getfield(L, cls_index, "__indices");
int indices_index = lua_gettop(L);
lua_getfield(L, cls_index, "__instances");
// Stack: [..., buckets]
immutable_info_t *info = (immutable_info_t*)lua_touserdata(L, 1); immutable_info_t *info = (immutable_info_t*)lua_touserdata(L, 1);
if (! info) { if (! info) {
luaL_error(L, "invalid type"); luaL_error(L, "invalid type");
} }
lua_rawgeti(L, -1, info->hash); lua_rawgeti(L, -1, info->hash);
// Stack: [..., buckets, bucket] // Stack: [mt, buckets, bucket]
if (lua_isnil(L, -1)) { if (lua_isnil(L, -1)) {
luaL_error(L, "Failed to find hash bucket"); luaL_error(L, "Failed to find hash bucket");
} }
lua_pushvalue(L, 1); lua_pushvalue(L, 1);
// Stack: [..., buckets, bucket, inst_udata] // Stack: [mt, buckets, bucket, inst_udata]
lua_rawget(L, -2); lua_rawget(L, -2);
// Stack: [..., buckets, bucket, inst_table] // Stack: [mt, buckets, bucket, inst_table]
size_t num_fields = lua_objlen(L, fields_index);
int index;
if (lua_isnil(L, 2)) {
index = 1;
} else {
lua_pushvalue(L, 2); lua_pushvalue(L, 2);
// Stack: [..., buckets, bucket, inst_table, k] // TODO: this is in a random order, and it might be good to force it to be in the same order as __fields
lua_gettable(L, indices_index); if (lua_next(L, -2) == 0) {
// Stack: [..., buckets, bucket, inst_table, i]
if (! lua_isnil(L, -1)) {
index = lua_tointeger(L, -1) + 1;
} else {
index = lua_tointeger(L, 2) + num_fields + 1;
}
lua_pop(L, 1);
}
// Stack: [..., buckets, bucket, inst_table]
if (! (0 < index && index <= (int)info->len)) {
return 0; return 0;
} }
if (lua_islightuserdata(L, -1) && lua_touserdata(L, -1) == &NIL_SENTINEL) {
// Now find field name lua_pop(L, 1);
if (index <= (int)num_fields) { lua_pushnil(L);
lua_rawgeti(L, fields_index, index);
} else {
lua_pushinteger(L, index - num_fields);
} }
// Stack: [..., buckets, bucket, inst_table, k2]
lua_rawgeti(L, -2, index);
// Stack: [..., buckets, bucket, inst_table, k2, value2]
return 2; return 2;
} }
@ -605,39 +584,18 @@ static int Lmake_class(lua_State *L)
// Stack: [CLS] // Stack: [CLS]
switch (num_args == 0 ? LUA_TNIL : lua_type(L, 1)) { switch (num_args == 0 ? LUA_TNIL : lua_type(L, 1)) {
case LUA_TTABLE: { case LUA_TTABLE:
// CLS.__fields = arg1
lua_pushvalue(L, 1); lua_pushvalue(L, 1);
// Stack: [CLS, __fields]
lua_setfield(L, -2, "__fields");
// Stack: [CLS]
size_t n = lua_objlen(L, 1);
lua_createtable(L, 0, n);
// Stack: [CLS, __indices]
lua_pushnil(L);
while (lua_next(L, 1) != 0) {
// Stack: [CLS, __indices, i, fieldname]
lua_pushvalue(L, -2);
// Stack: [CLS, __indices, i, fieldname, i]
lua_settable(L, -4);
// Stack: [CLS, __indices, i]
}
lua_setfield(L, -2, "__indices");
break; break;
} case LUA_TNIL: case LUA_TNONE:
case LUA_TNIL: case LUA_TNONE: { // If no fields were passed in, set __fields to empty table
// If no fields were passed in, set __fields and __indices to empty tables
lua_createtable(L, 0, 0); lua_createtable(L, 0, 0);
lua_setfield(L, -2, "__fields");
lua_createtable(L, 0, 0);
lua_setfield(L, -2, "__indices");
break; break;
}
default: { default: {
luaL_error(L, "immutable expected the fields to be either table or nil"); luaL_error(L, "immutable expected the fields to be either table or nil");
} }
} }
lua_setfield(L, -2, "__fields");
// Stack: [CLS] // Stack: [CLS]
lua_pushlightuserdata(L, (void*)&SHARED_CLASS_METATABLE); lua_pushlightuserdata(L, (void*)&SHARED_CLASS_METATABLE);
lua_gettable(L, LUA_REGISTRYINDEX); lua_gettable(L, LUA_REGISTRYINDEX);