Overhaul to remove duplicated code and ensure speedier constructor calls

for duplicate tables.
This commit is contained in:
Bruce Hill 2018-06-04 18:35:11 -07:00
parent a3fcf87b2c
commit 88ed3df58c
2 changed files with 130 additions and 264 deletions

View File

@ -50,202 +50,42 @@ 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;
typedef struct { typedef struct {
int hash; int hash;
size_t len; size_t len;
} immutable_info_t; } immutable_info_t;
static int Lcreate_instance(lua_State *L) #define GET(i) if (from_table) {\
{ if (i <= num_fields) {\
int num_args = lua_gettop(L)-1; lua_rawgeti(L, fields_index, i);\
lua_getfield(L, 1, "__fields"); } else {\
int num_fields = lua_objlen(L, -1); lua_pushinteger(L, i-num_fields);\
lua_pop(L, 1); }\
// arg 1: class table, ... lua_gettable(L, 2);\
lua_getfield(L, 1, "__new"); } else {\
if (! lua_isnil(L, -1)) { if (i <= num_args) {\
// stack: cls, ..., __new lua_pushvalue(L, 1+i);\
lua_pushvalue(L, 1); } else {\
// stack: cls, ..., __new, cls lua_pushnil(L);\
lua_rotate(L, 2, 2); }\
// stack: cls, __new, cls, ... }
lua_call(L, num_args+1, LUA_MULTRET);
num_args = lua_gettop(L)-1; static inline int _create_instance(lua_State *L, int from_table)
// stack: cls, ...
} else {
lua_pop(L, 1);
}
// Compute the hash:
unsigned long long ull_hash = 0x9a937c4d; // Seed
for (lua_Integer i=1; i <=(lua_Integer)num_fields || i <=(lua_Integer)num_args; i++) {
unsigned long long item_hash;
int type = i > num_args ? LUA_TNIL : lua_type(L, 1+i);
switch (type) {
case LUA_TNIL: case LUA_TNONE:
// Arbitrarily chosen value
item_hash = 0x97167da9;
break;
case LUA_TNUMBER:
{
// Cast float bits to integer
lua_Number num = lua_tonumber(L, 1+i);
item_hash = *((lua_Integer*)&num);
if (item_hash == 0) {
item_hash = 0x2887c992;
}
break;
}
case LUA_TBOOLEAN:
// Arbitrarily chosen values
item_hash = lua_toboolean(L, 1+i)? 0x82684f71 : 0x88d66f2a;
break;
case LUA_TTABLE:
case LUA_TFUNCTION:
case LUA_TUSERDATA:
case LUA_TTHREAD:
case LUA_TLIGHTUSERDATA:
item_hash = (1000003 * type) ^ (lua_Integer)lua_topointer(L, 1+i);
break;
case LUA_TSTRING:
{
// Algorithm taken from Lua 5.3's implementation
size_t len;
const char *str = lua_tolstring(L, 1+i, &len);
item_hash = len ^ 0xd2e9e9ac; // Arbitrary seed
size_t step = (len >> LUAI_HASHLIMIT) + 1;
for (; len >= step; len -= step)
item_hash ^= ((item_hash<<5) + (item_hash>>2) + (unsigned char)(str[len - 1]));
break;
}
default:
item_hash = 0;
}
ull_hash = (1000003 * ull_hash) ^ item_hash;
}
int hash = (int)ull_hash;
lua_getfield(L, 1, "__instances");
// Stack: [__instances]
// Find bucket
lua_rawgeti(L, -1, hash);
// Stack: [__instances, bucket]
if (lua_isnil(L, -1)) {
// Make a new bucket
// Stack: [__instances, nil]
lua_pop(L, 1);
// Stack: [__instances]
lua_createtable(L, 0, 1);
// Stack: [__instances, bucket]
lua_pushlightuserdata(L, (void*)&WEAK_KEY_METATABLE);
lua_gettable(L, LUA_REGISTRYINDEX);
// Stack: [__instances, bucket, {__mode='k'}]
lua_setmetatable(L, -2);
// Stack: [__instances, bucket]
lua_pushvalue(L, -1);
// Stack: [__instances, bucket, bucket]
lua_rawseti(L, -3, hash);
// Stack: [__instances, bucket]
} else {
// Stack: [__instances, bucket]
// scan bucket
lua_pushnil(L);
while (lua_next(L, -2) != 0) { // for hash_collider_inst, hash_collider in pairs(bucket) do
// Stack: [__instances, bucket, hash_collider_inst, hash_collider]
// Shallow equality check:
lua_pushnil(L);
while (lua_next(L, -2) != 0) { // for i, collider_value in pairs(hash_collider) do
// Stack: [__instances, 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
// Stack: [__instances, bucket, hash_collider_inst, hash_collider, i, value]
lua_pop(L, 3);
// Stack: [__instances, bucket, hash_collider_inst]
goto next_bucket_item;
} else {
// Stack: [__instances, bucket, hash_collider_inst, hash_collider, i, value]
lua_pop(L, 1);
// Stack: [__instances, bucket, hash_collider_inst, hash_collider, i]
}
}
// bucket item matches
// Stack: [__instances, bucket, hash_collider_inst, hash_collider]
lua_pop(L, 1);
return 1;
next_bucket_item: ;
}
}
lua_insert(L, -2);
// Stack: [bucket, __instances]
lua_pop(L, 1);
// Stack: [bucket]
int bucket_index = lua_gettop(L);
// Failed to find an existing instance, so create a new one
// Stack: [bucket]
immutable_info_t *userdata = (immutable_info_t*)lua_newuserdata(L, sizeof(immutable_info_t));
// Stack [bucket, inst_userdata]
int userdata_index = lua_gettop(L);
userdata->hash = hash;
userdata->len = num_fields > num_args ? num_fields : num_args;
lua_pushvalue(L, 1);
// Stack [bucket, inst_userdata, cls]
lua_setmetatable(L, -2);
// Stack [bucket, inst_userdata]
lua_createtable(L, userdata->len, 0); // Create the table to store the instance's data
// Stack [bucket, inst_userdata, inst_table]
for (lua_Integer i=1; i <= (lua_Integer)num_args; i++) {
lua_pushvalue(L, i+1);
// Stack [bucket, inst_userdata, inst_table, arg #1+i]
lua_rawseti(L, -2, i);
}
// Set up a ref to the bucket so its lifetime is tied to inst_userdata
lua_getfield(L, 1, "__buckets");
// Stack [bucket, inst_userdata, inst_table, __buckets]
lua_pushvalue(L, userdata_index);
// Stack [bucket, inst_userdata, inst_table, __buckets, inst_userdata]
lua_pushvalue(L, bucket_index);
// Stack [bucket, inst_userdata, inst_table, __buckets, inst_userdata, bucket]
lua_settable(L, -3);
lua_pop(L, 1);
// Stack [bucket, inst_userdata, inst_table]
lua_pushvalue(L, userdata_index);
// Stack [bucket, inst_userdata, inst_table, inst_userdata]
lua_rotate(L, bucket_index, 1);
// Stack [inst_userdata, bucket, inst_userdata, inst_table]
lua_settable(L, -3); // buckets[inst_userdata] = inst_table
lua_pop(L, 1);
// Stack [inst_userdata]
lua_gc(L, LUA_GCSTEP, 3);
return 1;
}
static int Lfrom_table(lua_State *L)
{ {
size_t num_args = lua_gettop(L)-1;
lua_getfield(L, 1, "__fields"); lua_getfield(L, 1, "__fields");
int fields_index = lua_gettop(L); int fields_index = lua_gettop(L);
int num_fields = lua_objlen(L, -1); size_t num_fields = lua_objlen(L, -1);
int num_values = num_fields + lua_objlen(L, 2); size_t num_values = from_table ? (num_fields + lua_objlen(L, 2)) : (num_args >= num_fields ? num_args : num_fields);
// Compute the hash: // Compute the hash and populate the values table:
unsigned long long ull_hash = 0x9a937c4d; // Seed
for (lua_Integer i=1; i <=(lua_Integer)num_values; i++) {
unsigned long long item_hash;
// Stack: [fields] // Stack: [fields]
if (i <= num_fields) { unsigned long long ull_hash = 0x9a937c4d; // Seed
lua_rawgeti(L, fields_index, i); for (size_t i=1; i <= num_values; i++) {
// Stack: [fields, fields[i]] unsigned long long item_hash;
lua_gettable(L, 2); GET(i);
// Stack: [fields, table[fields[i]]]
} else {
lua_rawgeti(L, 2, i - num_fields);
// Stack: [fields, table[i-num_fields]]
}
// Stack: [fields, value[i]] // Stack: [fields, value[i]]
int type = lua_type(L, -1); int type = lua_type(L, -1);
switch (type) { switch (type) {
@ -295,7 +135,6 @@ static int Lfrom_table(lua_State *L)
int hash = (int)ull_hash; int hash = (int)ull_hash;
lua_getfield(L, 1, "__instances"); lua_getfield(L, 1, "__instances");
// Stack: [fields, __instances] // Stack: [fields, __instances]
// Find bucket // Find bucket
lua_rawgeti(L, -1, hash); lua_rawgeti(L, -1, hash);
@ -320,91 +159,92 @@ static int Lfrom_table(lua_State *L)
// Stack: [fields, __instances, bucket] // Stack: [fields, __instances, bucket]
// scan bucket // scan bucket
lua_pushnil(L); lua_pushnil(L);
while (lua_next(L, -2) != 0) { // for hash_collider_inst, hash_collider in pairs(bucket) do while (lua_next(L, -2) != 0) { // for hash_collider_inst, collider_table in pairs(bucket) do
// Stack: [fields, __instances, bucket, hash_collider_inst, hash_collider] // Stack: [fields, __instances, bucket, hash_collider_inst, collider_table]
// Shallow equality check: // Shallow equality check:
lua_pushnil(L); immutable_info_t *collider_info = (immutable_info_t*)lua_touserdata(L, -2);
while (lua_next(L, -2) != 0) { // for i, collider_value in pairs(hash_collider) do if (collider_info->len != num_values) {
// Stack: [fields, __instances, bucket, hash_collider_inst, hash_collider, i, value] lua_pop(L, 1);
int i = lua_tonumber(L, -2); goto next_bucket_item;
if (i <= num_fields) {
lua_rawgeti(L, fields_index, i);
lua_gettable(L, 2);
} else {
lua_rawgeti(L, 2, i-num_fields);
} }
// Stack: [fields, __instances, bucket, hash_collider_inst, hash_collider, i, value, table_value] for (size_t i = 1; i <= num_values; i++) {
lua_rawgeti(L, -1, i);
GET(i);
// 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
// Stack: [fields, __instances, bucket, hash_collider_inst, hash_collider, i, value, table_value] lua_pop(L, 3);
lua_pop(L, 4);
// Stack: [fields, __instances, bucket, hash_collider_inst]
goto next_bucket_item; goto next_bucket_item;
} else { } else {
// Stack: [fields, __instances, bucket, hash_collider_inst, hash_collider, i, value, table_value]
lua_pop(L, 2); lua_pop(L, 2);
// Stack: [fields, __instances, bucket, hash_collider_inst, hash_collider, i]
} }
} }
// bucket item matches // bucket item matches
// Stack: [fields, __instances, bucket, hash_collider_inst, hash_collider] // Stack: [fields, __instances, bucket, hash_collider_inst, collider_table]
lua_pop(L, 1); lua_pop(L, 1);
return 1; return 1;
next_bucket_item: ; next_bucket_item: ;
} }
} }
lua_insert(L, -2); // Stack: [fields, __instances, bucket]
// Stack: [fields, bucket, __instances]
lua_pop(L, 1);
// Stack: [fields, bucket]
int bucket_index = lua_gettop(L); int bucket_index = lua_gettop(L);
// Failed to find an existing instance, so create a new one // Failed to find an existing instance, so create a new one
// Stack: [fields, bucket]
immutable_info_t *userdata = (immutable_info_t*)lua_newuserdata(L, sizeof(immutable_info_t)); immutable_info_t *userdata = (immutable_info_t*)lua_newuserdata(L, sizeof(immutable_info_t));
// Stack [fields, 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->len = num_values;
lua_pushvalue(L, 1); lua_pushvalue(L, 1);
// Stack [fields, bucket, inst_userdata, cls] // Stack: [fields, __instances, bucket, inst_userdata, cls]
lua_setmetatable(L, -2); lua_setmetatable(L, -2);
// Stack [fields, bucket, inst_userdata] // Stack: [fields, __instances, bucket, inst_userdata]
lua_createtable(L, userdata->len, 0); // Create the table to store the instance's data
// Stack [fields, bucket, inst_userdata, inst_table]
for (lua_Integer i=1; i <= (lua_Integer)num_fields; i++) {
lua_rawgeti(L, fields_index, i);
lua_gettable(L, 2);
lua_rawseti(L, -2, i);
}
for (lua_Integer i=num_fields+1; i <= num_values; i++) {
lua_rawgeti(L, 2, i-num_fields);
lua_rawseti(L, -2, i);
}
// Set up a ref to the bucket so its lifetime is tied to inst_userdata // Set up a ref to the bucket so its lifetime is tied to inst_userdata
lua_getfield(L, 1, "__buckets"); lua_getfield(L, 1, "__buckets");
// Stack [fields, bucket, inst_userdata, inst_table, __buckets] // Stack: [fields, __instances, bucket, inst_userdata, __buckets]
lua_pushvalue(L, userdata_index); lua_pushvalue(L, userdata_index);
// Stack [fields, bucket, inst_userdata, inst_table, __buckets, inst_userdata] // Stack: [fields, __instances, bucket, inst_userdata, __buckets, inst_userdata]
lua_pushvalue(L, bucket_index); lua_pushvalue(L, bucket_index);
// Stack [fields, bucket, inst_userdata, inst_table, __buckets, inst_userdata, bucket] // Stack: [fields, __instances, bucket, inst_userdata, __buckets, inst_userdata, bucket]
lua_settable(L, -3); lua_settable(L, -3);
lua_pop(L, 1); lua_pop(L, 1);
// Stack: [fields, __instances, bucket, inst_userdata]
// Stack [fields, bucket, inst_userdata, inst_table]
lua_pushvalue(L, userdata_index); lua_pushvalue(L, userdata_index);
// Stack [fields, bucket, inst_userdata, inst_table, inst_userdata] // Stack: [fields, __instances, bucket, inst_userdata, inst_userdata]
lua_rotate(L, bucket_index, 1); lua_createtable(L, num_values, 0);
// Stack [fields, inst_userdata, bucket, inst_userdata, inst_table] for (size_t i=1; i <= num_values; i++) {
lua_settable(L, -3); // buckets[inst_userdata] = inst_table GET(i);
lua_pop(L, 1); lua_rawseti(L, -2, i);
// Stack [fields, inst_userdata] }
// Stack: [fields, __instances, bucket, inst_userdata, inst_userdata, inst_table]
lua_settable(L, -4); // buckets[inst_userdata] = inst_table
// Stack: [fields, __instances, bucket, inst_userdata]
lua_gc(L, LUA_GCSTEP, 3); lua_gc(L, LUA_GCSTEP, 3);
return 1; return 1;
} }
static int Lcreate_instance(lua_State *L)
{
int num_args = lua_gettop(L)-1;
lua_getfield(L, 1, "__new");
if (! lua_isnil(L, -1)) {
lua_pushvalue(L, 1);
lua_rotate(L, 2, 2);
lua_call(L, num_args+1, LUA_MULTRET);
} else {
lua_pop(L, 1);
}
return _create_instance(L, 0);
}
static int Lfrom_table(lua_State *L)
{
return _create_instance(L, 1);
}
static int Lis_instance(lua_State *L) static int Lis_instance(lua_State *L)
{ {
if (lua_type(L, 2) != LUA_TUSERDATA) { if (lua_type(L, 2) != LUA_TUSERDATA) {
@ -490,10 +330,7 @@ static int Lindex(lua_State *L)
{ {
int ret = Linst_index(L); int ret = Linst_index(L);
if (ret > 0) { if (ret > 0) {
if (ret > 1) { return ret;
lua_pop(L, ret-1);
}
return 1;
} }
// Fall back to class: // Fall back to class:
lua_getmetatable(L, 1); lua_getmetatable(L, 1);
@ -512,7 +349,6 @@ static int Ltostring(lua_State *L)
{ {
luaL_Buffer b; luaL_Buffer b;
luaL_buffinit(L, &b); luaL_buffinit(L, &b);
if (! lua_getmetatable(L, 1)) { if (! lua_getmetatable(L, 1)) {
luaL_error(L, "invalid type"); luaL_error(L, "invalid type");
} }
@ -584,6 +420,7 @@ 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]
@ -592,7 +429,10 @@ static int Lnexti(lua_State *L)
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_isnil(L, 2) ? 1 : lua_tointeger(L, 2)+1;
if (i > (int)info->len) { lua_getfield(L, cls_index, "__fields");
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);
@ -606,11 +446,7 @@ 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_getfield(L, -5, "__fields"); lua_rawgeti(L, -2, i + num_fields);
// Stack: [mt, buckets, bucket, inst_table, i, __fields]
i += lua_objlen(L, -1);
lua_pop(L, 1);
lua_rawgeti(L, -2, i);
// Stack: [mt, buckets, bucket, inst_table, i, table[i]] // Stack: [mt, buckets, bucket, inst_table, i, table[i]]
return 2; return 2;
} }
@ -707,15 +543,10 @@ static int Lhash(lua_State *L)
static const luaL_Reg Rinstance_metamethods[] = static const luaL_Reg Rinstance_metamethods[] =
{ {
{"__len", Llen}, {"__len", Llen},
{"__rawlen", Llen},
{"__index", Lindex}, {"__index", Lindex},
{"__rawindex", Linst_index},
{"__tostring", Ltostring}, {"__tostring", Ltostring},
{"__rawtostring", Ltostring},
{"__ipairs", Lipairs}, {"__ipairs", Lipairs},
{"__rawipairs", Lipairs},
{"__pairs", Lpairs}, {"__pairs", Lpairs},
{"__rawpairs", Lpairs},
{"__hash", Lhash}, {"__hash", Lhash},
{"from_table", Lfrom_table}, {"from_table", Lfrom_table},
{"is_instance", Lis_instance}, {"is_instance", Lis_instance},
@ -748,6 +579,10 @@ static int Lmake_class(lua_State *L)
// Stack: [CLS] // Stack: [CLS]
} }
lua_pushlightuserdata(L, (void*)&SUPER_METHODS);
lua_gettable(L, LUA_REGISTRYINDEX);
lua_setfield(L, -2, "__super");
// Stack: [CLS] // Stack: [CLS]
lua_createtable(L, 0, 32); // Rough guess: at least 32 instances concurrently lua_createtable(L, 0, 32); // Rough guess: at least 32 instances concurrently
// Stack: [CLS, __instances] // Stack: [CLS, __instances]
@ -830,6 +665,11 @@ static const luaL_Reg Rclass_metamethods[] =
LUALIB_API int luaopen_immutable(lua_State *L) LUALIB_API int luaopen_immutable(lua_State *L)
{ {
lua_pushlightuserdata(L, (void*)&SUPER_METHODS);
lua_createtable(L, 0, 8);
luaL_register(L,NULL,Rinstance_metamethods);
lua_settable(L, LUA_REGISTRYINDEX);
lua_pushlightuserdata(L, (void*)&WEAK_VALUE_METATABLE); lua_pushlightuserdata(L, (void*)&WEAK_VALUE_METATABLE);
lua_createtable(L, 0, 1); lua_createtable(L, 0, 1);
lua_pushstring(L, "v"); lua_pushstring(L, "v");

View File

@ -31,6 +31,7 @@ local function test(description, fn)
io.write("\r"..description.."\n"..reset) io.write("\r"..description.."\n"..reset)
print(reset..red..(err or "")..reset) print(reset..red..(err or "")..reset)
num_errors = num_errors + 1 num_errors = num_errors + 1
--os.exit(true)
else else
io.write(reset..dim.."\r.......................................") io.write(reset..dim.."\r.......................................")
io.write(reset.."["..green.."PASSED"..reset.."]\r") io.write(reset.."["..green.."PASSED"..reset.."]\r")
@ -45,6 +46,7 @@ end)
test("Creating class", function() test("Creating class", function()
Vec = immutable({"x","y"}, { Vec = immutable({"x","y"}, {
name="Vec",
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,
@ -52,9 +54,6 @@ test("Creating class", function()
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)
end, end,
__tostring=function(self)
return "Vec("..tostring(self.x)..", "..tostring(self.y)..")"
end,
}) })
end) end)
@ -72,6 +71,10 @@ test("Testing # operator", function()
assert(#T(1,2,3) == 3) assert(#T(1,2,3) == 3)
end) end)
test("Testing # operator for mixed table", function()
assert(#Vec(1,2,3,4,5) == 3)
end)
test("Testing method", function() test("Testing method", function()
assert(v:len2() == 10) assert(v:len2() == 10)
end) end)
@ -81,10 +84,10 @@ test("Testing tostring", function()
end) end)
test("Testing from_table", function() test("Testing from_table", function()
assert(Vec:from_table({x=1,y=2}) == Vec(1,2)) assert(Vec:from_table({x=1,y=2}) == Vec(1,2), "assertion 1 failed!")
assert(Vec:from_table({3, x=1,y=2}) == Vec(1,2,3)) assert(Vec:from_table({3, x=1,y=2}) == Vec(1,2,3), "assertion 2 failed!")
assert(Vec:from_table(setmetatable({y=3}, {__index={x=1}})) == v) assert(Vec:from_table(setmetatable({y=3}, {__index={x=1}})) == v, "assertion 3 failed!")
assert(Vec:from_table({x=1}) == Vec(1, nil)) assert(Vec:from_table({x=1}) == Vec(1, nil), "assertion 4 failed!")
end) end)
test("Testing from_table for tuples", function() test("Testing from_table for tuples", function()
@ -175,9 +178,9 @@ test("Testing __rawindex", function()
classvar = 99, classvar = 99,
__index=function(self,k) __index=function(self,k)
local cls = getmetatable(self) local cls = getmetatable(self)
local inst_val, found = cls.__rawindex(self, k) local inst_val, found = cls.__super.__index(self, k)
if found then return inst_val end if found then return inst_val end
return cls[k] or tostring(k)..tostring(self.x) return cls[k] or tostring(k)..tostring(cls.__super.__index(self, 'x'))
end, end,
}) })
local f = Foo(23, nil) local f = Foo(23, nil)
@ -260,31 +263,54 @@ if _VERSION == "Lua 5.2" or _VERSION == "Lua 5.3" then
local T = immutable() local T = immutable()
local t = T(1,4,9,nil,16,nil) local t = T(1,4,9,nil,16,nil)
local checks = {1,4,9,nil,16,nil} local checks = {1,4,9,nil,16,nil}
local passed = 0 local passed, iterations = 0, 0
for i,v in ipairs(t) do for i,v in ipairs(t) do
assert(checks[i] == v) assert(checks[i] == v)
passed = passed + 1 passed = passed + 1
iterations = iterations + 1
end end
assert(passed == 6) assert(passed == 6 and iterations == passed)
passed = 0 passed, iterations = 0, 0
for k,v in pairs(t) do for k,v in pairs(t) do
assert(checks[k] == v) assert(checks[k] == v)
passed = passed + 1 passed = passed + 1
iterations = iterations + 1
end end
assert(passed == 6) assert(passed == 6 and iterations == passed)
end) end)
test("Testing table iteration", function() test("Testing table iteration", function()
local Foo = immutable({"x", "y", "z","w"}) local Foo = immutable({"x", "y", "z","w"})
local f = Foo(1,nil,2,nil) local f = Foo(1,nil,2,nil)
local checks = {1,nil,2,nil} local checks = {1,nil,2,nil}
local passed = 0 local passed, iterations = 0, 0
checks = {x=1,y=nil,z=2,w=nil} checks = {x=1,y=nil,z=2,w=nil}
for k,v in pairs(f) do for k,v in pairs(f) do
assert(checks[k] == v) assert(checks[k] == v)
passed = passed + 1 passed = passed + 1
iterations = iterations + 1
end end
assert(passed == 4) assert(passed == 4 and iterations == passed)
end)
test("Testing mixed iteration", function()
local T = immutable({"skip"})
local t = T('a','b','c')
local checks = {'b','c'}
local passed, iterations = 0, 0
for i,v in ipairs(t) do
assert(checks[i] == v, "checks["..i.."] ~= "..v)
passed = passed + 1
iterations = iterations + 1
end
assert(passed == 2 and iterations == passed, "ipairs failed")
checks = {skip='a','b','c'}
passed = 0
for k,v in pairs(t) do
assert(checks[k] == v, "checks["..k.."] ~= "..v)
passed = passed + 1
end
assert(passed == 3, "pairs failed")
end) end)
end end
@ -331,7 +357,7 @@ test("Testing __rawindex", function()
classvar = 99, classvar = 99,
__index = function(self, key) __index = function(self, key)
local cls = getmetatable(self) local cls = getmetatable(self)
local value, found = cls.__rawindex(self, key) local value, found = cls.__super.__index(self, key)
if not found then if not found then
return cls[key] return cls[key]
else else