Fixed some severe memory problems with slightly kludgey code. Classes

now have a weak-keyed __buckets field that maps udata->bucket, and
buckets now get auto-collected by GC. __gc metamethod is now gone.
This commit is contained in:
Bruce Hill 2018-02-12 21:59:01 -08:00
parent 565250d326
commit 45d410d00c
2 changed files with 103 additions and 80 deletions

View File

@ -28,7 +28,10 @@
// This is used to create a unique light userdata to store in the registry to allow all
// hash buckets to share the same {__mode='k'} metatable
static int SHARED_BUCKET_METATABLE;
static int WEAK_KEY_METATABLE;
// This is used to create a unique light userdata to store in the registry to allow all
// __instances tables to share the same {__mode='v'} metatable
static int WEAK_VALUE_METATABLE;
// This is used to create a unique light userdata to store in the registry to allow all
// constructed classes to have the same metatable:
// {__new=Lcreate_instance, __tostring=function(cls) return cls.name or 'immutable(...)' end}
@ -96,16 +99,18 @@ static int Lcreate_instance(lua_State *L)
// Find bucket
lua_rawgeti(L, -1, hash);
// Stack: [buckets, bucket]
int created_bucket = 0;
if (lua_isnil(L, -1)) {
created_bucket = 1;
// Make a new bucket
// Stack: [buckets, nil]
lua_pop(L, 1);
// Stack: [buckets]
lua_createtable(L, 0, 1);
// Stack: [buckets, bucket]
lua_pushlightuserdata(L, (void*)&SHARED_BUCKET_METATABLE);
lua_pushlightuserdata(L, (void*)&WEAK_KEY_METATABLE);
lua_gettable(L, LUA_REGISTRYINDEX);
// Stack: [buckets, bucket, {'__mode'='k'}]
// Stack: [buckets, bucket, {__mode='k'}]
lua_setmetatable(L, -2);
// Stack: [buckets, bucket]
lua_pushvalue(L, -1);
@ -114,32 +119,34 @@ static int Lcreate_instance(lua_State *L)
// Stack: [buckets, bucket]
}
// Stack: [buckets, bucket]
// scan bucket
lua_pushnil(L);
while (lua_next(L, -2) != 0) { // for hash_collider_inst, hash_collider in pairs(bucket) do
// Stack: [buckets, bucket, hash_collider_inst, hash_collider]
int bucket_item_matches = 1;
// Shallow equality check:
if (! created_bucket) {
// scan bucket
lua_pushnil(L);
while (lua_next(L, -2) != 0) { // for i, collider_value in pairs(hash_collider) do
// Stack: [buckets, bucket, hash_collider_inst, hash_collider, i, value]
if (!lua_rawequal(L, -1, 1+lua_tonumber(L, -2))) { // If the i'th entry doesn't match the i'th arg
bucket_item_matches = 0;
// Stack: [buckets, bucket, hash_collider_inst, hash_collider, i, value]
lua_pop(L, 3);
// Stack: [buckets, bucket, hash_collider_inst]
break; // go to next item in the bucket
} else {
// Stack: [buckets, bucket, hash_collider_inst, hash_collider, i, value]
lua_pop(L, 1);
// Stack: [buckets, bucket, hash_collider_inst, hash_collider, i]
}
}
if (bucket_item_matches) {
while (lua_next(L, -2) != 0) { // for hash_collider_inst, hash_collider in pairs(bucket) do
// Stack: [buckets, bucket, hash_collider_inst, hash_collider]
lua_pop(L, 1);
// Found matching pre-existing instance
return 1;
int bucket_item_matches = 1;
// Shallow equality check:
lua_pushnil(L);
while (lua_next(L, -2) != 0) { // for i, collider_value in pairs(hash_collider) do
// Stack: [buckets, bucket, hash_collider_inst, hash_collider, i, value]
if (!lua_rawequal(L, -1, 1+lua_tonumber(L, -2))) { // If the i'th entry doesn't match the i'th arg
bucket_item_matches = 0;
// Stack: [buckets, bucket, hash_collider_inst, hash_collider, i, value]
lua_pop(L, 3);
// Stack: [buckets, bucket, hash_collider_inst]
break; // go to next item in the bucket
} else {
// Stack: [buckets, bucket, hash_collider_inst, hash_collider, i, value]
lua_pop(L, 1);
// Stack: [buckets, bucket, hash_collider_inst, hash_collider, i]
}
}
if (bucket_item_matches) {
// Stack: [buckets, bucket, hash_collider_inst, hash_collider]
lua_pop(L, 1);
// Found matching pre-existing instance
return 1;
}
}
}
@ -155,7 +162,19 @@ static int Lcreate_instance(lua_State *L)
// Stack [buckets, bucket, inst_userdata]
lua_pushvalue(L, -1);
// Stack [buckets, bucket, inst_userdata, inst_userdata]
lua_createtable(L, n, 0); // Create the table to store the instance's data
lua_createtable(L, n, 1); // Create the table to store the instance's data
// Stack [buckets, bucket, inst_userdata, inst_userdata, inst_table]
// Set up a ref to the bucket so its lifetime is tied to inst_userdata
lua_getfield(L, 1, "__buckets");
// Stack [buckets, bucket, inst_userdata, inst_userdata, inst_table, __buckets]
lua_pushvalue(L, -3);
// Stack [buckets, bucket, inst_userdata, inst_userdata, inst_table, __buckets, inst_userdata]
lua_pushvalue(L, -6);
// Stack [buckets, bucket, inst_userdata, inst_userdata, inst_table, __buckets, inst_userdata, bucket]
lua_settable(L, -3);
lua_pop(L, 1);
// Stack [buckets, bucket, inst_userdata, inst_userdata, inst_table]
lua_Integer i;
for (i=1; i <= (lua_Integer)n_args; i++) {
@ -251,6 +270,9 @@ static int Lindex(lua_State *L)
}
lua_rawgeti(L, -1, *hash_address);
// Stack: [mt, indices, i, buckets, bucket]
if (lua_isnil(L, -1)) {
luaL_error(L, "Failed to find hash bucket for hash: %p", (void*)*hash_address);
}
lua_pushvalue(L, 1);
// Stack: [mt, indices, i, buckets, bucket, inst_udata]
lua_rawget(L, -2);
@ -268,6 +290,9 @@ static int Lindex(lua_State *L)
}
lua_rawgeti(L, -1, *hash_address);
// Stack: [mt, buckets, bucket]
if (lua_isnil(L, -1)) {
luaL_error(L, "Failed to find hash bucket");
}
lua_pushvalue(L, 1);
// Stack: [mt, buckets, bucket, inst_udata]
lua_rawget(L, -2);
@ -322,6 +347,9 @@ static int Ltostring(lua_State *L)
}
lua_rawgeti(L, -1, *hash_address);
// Stack: [mt, buckets, bucket]
if (lua_isnil(L, -1)) {
luaL_error(L, "Failed to find hash bucket");
}
lua_pushvalue(L, 1);
// Stack: [mt, buckets, bucket, inst_udata]
lua_rawget(L, -2);
@ -389,6 +417,9 @@ static int Lnexti(lua_State *L)
}
lua_rawgeti(L, -1, *hash_address);
// Stack: [mt, buckets, bucket]
if (lua_isnil(L, -1)) {
luaL_error(L, "Failed to find hash bucket");
}
lua_pushvalue(L, 1);
// Stack: [mt, buckets, bucket, inst_udata]
lua_rawget(L, -2);
@ -433,6 +464,9 @@ static int Lnext(lua_State *L)
}
lua_rawgeti(L, -1, *hash_address);
// Stack: [mt, buckets, bucket]
if (lua_isnil(L, -1)) {
luaL_error(L, "Failed to find hash bucket");
}
lua_pushvalue(L, 1);
// Stack: [mt, buckets, bucket, inst_udata]
lua_rawget(L, -2);
@ -461,45 +495,6 @@ static int Lpairs(lua_State *L)
return 3;
}
static int Lgc(lua_State *L)
{
// Unfortunately, a __gc metamethod is needed here, because the data storage format
// is cls.__instances[hash][udata] = data, and even though cls.__instances[hash]
// is a weak-keyed table, cls.__instances will leak hash key -> empty table mappings
// over time.
lua_getmetatable(L, 1);
// Stack: [mt]
lua_getfield(L, -1, "__instances");
// Stack: [mt, buckets]
lua_Integer* hash_address = (lua_Integer*)lua_touserdata(L, 1);
if (! hash_address) {
return 0;
}
lua_rawgeti(L, -1, *hash_address);
// Stack: [mt, indices, i, buckets, bucket]
if (lua_isnil(L, -1)) {
return 0;
}
// Set cls.__instances[hash][udata] = nil
lua_pushvalue(L, 1);
// Stack: [mt, indices, i, buckets, bucket, inst_udata]
lua_pushnil(L);
// Stack: [mt, indices, i, buckets, bucket, inst_udata, nil]
lua_settable(L, -3);
// Stack: [mt, indices, i, buckets, bucket]
lua_pushnil(L);
// If next(cls.__instances[hash]) == nil, then set cls.__instances[hash] = nil
// Stack: [mt, indices, i, buckets, bucket, nil]
if (lua_next(L, -2) == 0) {
lua_pop(L, 1);
// Stack: [mt, indices, i, buckets]
lua_pushnil(L);
// Stack: [mt, indices, i, buckets, nil]
lua_rawseti(L, -2, *hash_address);
}
return 1;
}
static const luaL_Reg Rinstance_metamethods[] =
{
{ "__len", Llen},
@ -507,7 +502,6 @@ static const luaL_Reg Rinstance_metamethods[] =
{ "__tostring", Ltostring},
{ "__ipairs", Lipairs},
{ "__pairs", Lpairs},
{ "__gc", Lgc},
{ "from_table", Lfrom_table},
{ "is_instance", Lis_instance},
{ NULL, NULL}
@ -542,9 +536,24 @@ static int Lmake_class(lua_State *L)
// Stack: [CLS]
lua_createtable(L, 0, 32); // Rough guess: at least 32 instances concurrently
// Stack: [CLS, CLS.buckets]
// Stack: [CLS, __instances]
lua_pushlightuserdata(L, (void*)&WEAK_VALUE_METATABLE);
lua_gettable(L, LUA_REGISTRYINDEX);
// Stack: [CLS, __instances, {__mode='v'}]
lua_setmetatable(L, -2);
// Stack: [CLS, __instances]
lua_setfield(L, -2, "__instances");
// Stack: [CLS]
lua_createtable(L, 0, 32); // Rough guess: at least 32 instances concurrently
// Stack: [CLS, __buckets]
lua_pushlightuserdata(L, (void*)&WEAK_KEY_METATABLE);
lua_gettable(L, LUA_REGISTRYINDEX);
// Stack: [CLS, __buckets, {__mode='k'}]
lua_setmetatable(L, -2);
// Stack: [CLS, __buckets]
lua_setfield(L, -2, "__buckets");
// Stack: [CLS]
switch (lua_type(L, 1)) {
case LUA_TTABLE: {
@ -632,7 +641,13 @@ static const luaL_Reg Rclass_metamethods[] =
LUALIB_API int luaopen_immutable(lua_State *L)
{
lua_pushlightuserdata(L, (void*)&SHARED_BUCKET_METATABLE);
lua_pushlightuserdata(L, (void*)&WEAK_VALUE_METATABLE);
lua_createtable(L, 0, 1);
lua_pushstring(L, "v");
lua_setfield(L, -2, "__mode");
lua_settable(L, LUA_REGISTRYINDEX);
lua_pushlightuserdata(L, (void*)&WEAK_KEY_METATABLE);
lua_createtable(L, 0, 1);
lua_pushstring(L, "k");
lua_setfield(L, -2, "__mode");

View File

@ -61,6 +61,10 @@ test("Instantiating class", function()
v = assert(Vec(1,3))
end)
test("Testing indexing", function()
assert(v.x == 1)
end)
test("Testing # operator", function()
assert(#v == 2)
end)
@ -117,28 +121,32 @@ collectgarbage()
collectgarbage()
test("Testing garbage collection", function()
collectgarbage()
collectgarbage()
local Foo = immutable({"x"})
local function countFoos()
collectgarbage()
collectgarbage()
local n = 0
local buckets = 0
for h,bucket in pairs(Foo.__instances) do
buckets = buckets + 1
for _ in pairs(bucket) do
n = n + 1
end
end
return n
collectgarbage()
collectgarbage()
return n, buckets
end
local f1, f2, also_f2 = Foo(1), Foo(2), Foo(2)
assert(countFoos() == 2)
local foos, buckets = countFoos()
assert(foos == 2, "WTF? "..tostring(foos))
f1, f2 = nil, nil
collectgarbage()
collectgarbage()
assert(countFoos() == 1)
foos, buckets = countFoos()
assert(foos == 1)
also_f2 = nil
collectgarbage()
collectgarbage()
assert(countFoos() == 0)
foos, buckets = countFoos()
assert(foos == 0, "Leaking instances")
assert(buckets == 0, "Leaking hash buckets")
assert(next(Foo.__instances) == nil)
end)