Cleanup and compressing methods/metamethods into the same thing.
This commit is contained in:
parent
468b6a83bb
commit
b58bd5a12e
53
limmutable.c
53
limmutable.c
@ -100,15 +100,11 @@ static int Lcreate_instance(lua_State *L)
|
|||||||
// Stack: [inst, buckets]
|
// Stack: [inst, buckets]
|
||||||
lua_createtable(L, 1, 0);
|
lua_createtable(L, 1, 0);
|
||||||
// Stack: [inst, buckets, bucket]
|
// Stack: [inst, buckets, bucket]
|
||||||
// TODO: share bucket metatables
|
lua_pushlightuserdata(L, (void*)Lcreate_instance);
|
||||||
lua_createtable(L, 0, 1);
|
lua_gettable(L, LUA_REGISTRYINDEX);
|
||||||
// Stack: [inst, buckets, bucket, bucket_mt]
|
|
||||||
lua_pushstring(L, "k");
|
|
||||||
lua_setfield(L, -2, "__mode");
|
|
||||||
// Stack: [inst, buckets, bucket, {'__mode'='k'}]
|
// Stack: [inst, buckets, bucket, {'__mode'='k'}]
|
||||||
lua_setmetatable(L, -2);
|
lua_setmetatable(L, -2);
|
||||||
// Stack: [inst, buckets, bucket]
|
// Stack: [inst, buckets, bucket]
|
||||||
|
|
||||||
lua_pushvalue(L, -1);
|
lua_pushvalue(L, -1);
|
||||||
// Stack: [inst, buckets, bucket, bucket]
|
// Stack: [inst, buckets, bucket, bucket]
|
||||||
lua_rawseti(L, -3, hash);
|
lua_rawseti(L, -3, hash);
|
||||||
@ -376,6 +372,7 @@ static const luaL_Reg R[] =
|
|||||||
{
|
{
|
||||||
{ "__len", Llen},
|
{ "__len", Llen},
|
||||||
{ "__index", Lindex},
|
{ "__index", Lindex},
|
||||||
|
{ "__tostring", Ltostring},
|
||||||
{ "__ipairs", Lipairs},
|
{ "__ipairs", Lipairs},
|
||||||
{ "__pairs", Lpairs},
|
{ "__pairs", Lpairs},
|
||||||
{ "from_table", Lfrom_table},
|
{ "from_table", Lfrom_table},
|
||||||
@ -384,21 +381,24 @@ static const luaL_Reg R[] =
|
|||||||
|
|
||||||
static int Lmake_class(lua_State *L)
|
static int Lmake_class(lua_State *L)
|
||||||
{
|
{
|
||||||
// args: fields, [methods, [metamethods]]
|
// args: fields, [methods/metamethods]
|
||||||
int n_args = lua_gettop(L);
|
int n_args = lua_gettop(L);
|
||||||
|
|
||||||
// CLS = {}
|
// CLS = {}
|
||||||
lua_newtable(L);
|
lua_newtable(L);
|
||||||
// Stack: [CLS]
|
// Stack: [CLS]
|
||||||
lua_pushcfunction(L, Ltostring);
|
// Populate CLS.__len, CLS.__index, CLS.__pairs, etc.
|
||||||
lua_setfield(L, -2, "__tostring");
|
luaL_register(L,NULL,R);
|
||||||
|
|
||||||
// If metamethods were passed in, copy them over
|
// If methods were passed in, copy them over, overwriting defaults if desired
|
||||||
if (n_args > 2) {
|
if (n_args >= 2) {
|
||||||
|
// Stack: [CLS]
|
||||||
lua_pushnil(L);
|
lua_pushnil(L);
|
||||||
// Stack: [CLS, nil]
|
// Stack: [CLS, nil]
|
||||||
while (lua_next(L, 3) != 0) {
|
while (lua_next(L, 2) != 0) {
|
||||||
|
// Stack: [CLS, method_name, method_value]
|
||||||
lua_pushvalue(L, -2);
|
lua_pushvalue(L, -2);
|
||||||
|
// Stack: [CLS, method_name, method_value, method_name]
|
||||||
lua_pushvalue(L, -2);
|
lua_pushvalue(L, -2);
|
||||||
// Stack: [CLS, method_name, method_value, method_name, method_value]
|
// Stack: [CLS, method_name, method_value, method_name, method_value]
|
||||||
lua_settable(L, -5);
|
lua_settable(L, -5);
|
||||||
@ -406,9 +406,10 @@ static int Lmake_class(lua_State *L)
|
|||||||
lua_pop(L, 1);
|
lua_pop(L, 1);
|
||||||
// Stack: [CLS, method_name]
|
// Stack: [CLS, method_name]
|
||||||
}
|
}
|
||||||
|
// Stack: [CLS]
|
||||||
}
|
}
|
||||||
|
|
||||||
// CLS.buckets = {}
|
// Stack: [CLS]
|
||||||
lua_newtable(L);
|
lua_newtable(L);
|
||||||
// Stack: [CLS, CLS.buckets]
|
// Stack: [CLS, CLS.buckets]
|
||||||
lua_setfield(L, -2, "__instances");
|
lua_setfield(L, -2, "__instances");
|
||||||
@ -435,27 +436,6 @@ static int Lmake_class(lua_State *L)
|
|||||||
lua_setfield(L, -2, "__indices");
|
lua_setfield(L, -2, "__indices");
|
||||||
// Stack: [CLS]
|
// Stack: [CLS]
|
||||||
|
|
||||||
// If methods were passed in, copy them over
|
|
||||||
if (n_args > 1) {
|
|
||||||
// Stack: [
|
|
||||||
lua_pushnil(L);
|
|
||||||
// Stack: [CLS, nil]
|
|
||||||
while (lua_next(L, 2) != 0) {
|
|
||||||
// Stack: [CLS, method_name, method_value]
|
|
||||||
lua_pushvalue(L, -2);
|
|
||||||
lua_pushvalue(L, -2);
|
|
||||||
// Stack: [CLS, method_name, method_value, method_name, method_value]
|
|
||||||
lua_settable(L, -5);
|
|
||||||
// Stack: [CLS, method_name, method_value]
|
|
||||||
lua_pop(L, 1);
|
|
||||||
// Stack: [CLS, method_name]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Stack: [CLS]
|
|
||||||
|
|
||||||
// Populate __len, __eq, __index, new, etc.
|
|
||||||
luaL_register(L,NULL,R);
|
|
||||||
|
|
||||||
// setmetatable(CLS, {__new=CLS.new})
|
// setmetatable(CLS, {__new=CLS.new})
|
||||||
lua_createtable(L, 0, 1);
|
lua_createtable(L, 0, 1);
|
||||||
lua_pushcfunction(L, Lcreate_instance);
|
lua_pushcfunction(L, Lcreate_instance);
|
||||||
@ -468,6 +448,11 @@ static int Lmake_class(lua_State *L)
|
|||||||
|
|
||||||
LUALIB_API int luaopen_immutable(lua_State *L)
|
LUALIB_API int luaopen_immutable(lua_State *L)
|
||||||
{
|
{
|
||||||
|
lua_pushlightuserdata(L, (void*)Lcreate_instance);
|
||||||
|
lua_createtable(L, 0, 1);
|
||||||
|
lua_pushstring(L, "k");
|
||||||
|
lua_setfield(L, -2, "__mode");
|
||||||
|
lua_settable(L, LUA_REGISTRYINDEX);
|
||||||
lua_pushcfunction(L, Lmake_class);
|
lua_pushcfunction(L, Lmake_class);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
20
test.lua
20
test.lua
@ -47,7 +47,6 @@ test("Creating class", function()
|
|||||||
len2=function(self)
|
len2=function(self)
|
||||||
return self.x*self.x + self.y*self.y
|
return self.x*self.x + self.y*self.y
|
||||||
end,
|
end,
|
||||||
}, {
|
|
||||||
__add=function(self, other)
|
__add=function(self, other)
|
||||||
local cls = getmetatable(self)
|
local cls = getmetatable(self)
|
||||||
return cls(self.x+other.x, self.y+other.y)
|
return cls(self.x+other.x, self.y+other.y)
|
||||||
@ -110,12 +109,29 @@ end)
|
|||||||
collectgarbage()
|
collectgarbage()
|
||||||
collectgarbage()
|
collectgarbage()
|
||||||
|
|
||||||
|
test("Testing garbage collection", function()
|
||||||
|
local collected = false
|
||||||
|
local GCSnooper = immutable({}, {
|
||||||
|
__gc=function(self)
|
||||||
|
collected = true
|
||||||
|
end,
|
||||||
|
})
|
||||||
|
local g = GCSnooper()
|
||||||
|
collectgarbage()
|
||||||
|
collectgarbage()
|
||||||
|
assert(not collected)
|
||||||
|
g = nil
|
||||||
|
collectgarbage()
|
||||||
|
collectgarbage()
|
||||||
|
assert(collected)
|
||||||
|
end)
|
||||||
|
|
||||||
|
|
||||||
test("Testing similar class", function()
|
test("Testing similar class", function()
|
||||||
local FooVec = immutable({"x","y"}, {
|
local FooVec = immutable({"x","y"}, {
|
||||||
len2=function(self)
|
len2=function(self)
|
||||||
return self.x*self.x + self.y*self.y
|
return self.x*self.x + self.y*self.y
|
||||||
end,
|
end,
|
||||||
}, {
|
|
||||||
__add=function(self, other)
|
__add=function(self, other)
|
||||||
local cls = getmetatable(self)
|
local cls = getmetatable(self)
|
||||||
return cls(self.x+other.x, self.y+other.y)
|
return cls(self.x+other.x, self.y+other.y)
|
||||||
|
Loading…
Reference in New Issue
Block a user