diff --git a/VM/src/lgc.cpp b/VM/src/lgc.cpp
index f7a851f4f..6e8dfd20c 100644
--- a/VM/src/lgc.cpp
+++ b/VM/src/lgc.cpp
@@ -15,6 +15,8 @@
#define GC_SWEEPPAGESTEPCOST 16
+static constexpr int kLastElement = 0x100;
+
#define GC_INTERRUPT(state) \
{ \
void (*interrupt)(lua_State*, int) = g->cb.interrupt; \
@@ -31,17 +33,17 @@
#define stringmark(s) reset2bits((s)->marked, WHITE0BIT, WHITE1BIT)
-#define markvalue(g, o) \
+#define markvalue(A, g, o) \
{ \
checkconsistency(o); \
if (iscollectable(o) && iswhite(gcvalue(o))) \
- reallymarkobject(g, gcvalue(o)); \
+ reallymarkobject(g, gcvalue(o)); \
}
-#define markobject(g, t) \
+#define markobject(A, g, t) \
{ \
if (iswhite(obj2gco(t))) \
- reallymarkobject(g, obj2gco(t)); \
+ reallymarkobject(g, obj2gco(t)); \
}
#ifdef LUAI_GCMETRICS
@@ -128,9 +130,14 @@ static void removeentry(LuaNode* n)
setttype(gkey(n), LUA_TDEADKEY); // dead key; remove it
}
-static void reallymarkobject(global_State* g, GCObject* o)
+template
+static void reallymarkobject(global_State* g, GCObject* o);
+
+template
+static void reallymarkobject0(global_State* g, GCObject* o)
{
LUAU_ASSERT(iswhite(o) && !isdead(g, o));
+ LUAU_ASSERT(!A || !isephemeronkey(o));
white2gray(o);
switch (o->gch.tt)
{
@@ -143,13 +150,13 @@ static void reallymarkobject(global_State* g, GCObject* o)
Table* mt = gco2u(o)->metatable;
gray2black(o); // udata are never gray
if (mt)
- markobject(g, mt);
+ markobject(A, g, mt);
return;
}
case LUA_TUPVAL:
{
UpVal* uv = gco2uv(o);
- markvalue(g, uv->v);
+ markvalue(A, g, uv->v);
if (uv->v == &uv->u.value) // closed?
gray2black(o); // open upvalues are never black
return;
@@ -183,6 +190,170 @@ static void reallymarkobject(global_State* g, GCObject* o)
}
}
+
+static void** getephemeronlink(GCObject* o) {
+ // TODO: These casts are not nice and could be done better (with unions)
+ switch (o->gch.tt)
+ {
+ case LUA_TUSERDATA:
+ {
+ return reinterpret_cast(&gco2u(o)->metatable);
+ }
+ case LUA_TFUNCTION:
+ {
+ return reinterpret_cast(&gco2cl(o)->gclist);
+ }
+ case LUA_TTABLE:
+ {
+ return reinterpret_cast(&gco2h(o)->gclist);
+ }
+ case LUA_TTHREAD:
+ {
+ return reinterpret_cast(&gco2th(o)->gclist);
+ }
+ default:
+ LUAU_ASSERT(0);
+ return nullptr;
+ }
+}
+
+/**
+ * @brief Link a table node from the ephemeron list to the worklist.
+ *
+ * @param o Object to restore the key.
+ * @param n Table node
+ * @param list Worklist
+ * @return LuaNode* Adapted worklist
+ */
+static LuaNode* relink(GCObject* o, LuaNode* n, LuaNode* list) {
+ uintptr_t intptr = reinterpret_cast(list);
+ ttype(gkey(n)) = gkey(n)->extra[0] & 0xF; // Restore value tt and save it in the key tt.
+ gkey(n)->value.gc = o; // Restore the key value, key tt needs to be restored later.
+ gkey(n)->extra[0] = static_cast(static_cast(intptr & 0xFFFFFFFF));
+ // Save the link to the next worklist entry in the key extra slot and value tt slot.
+ // Note: The value tt is saved in the key tt which can be restored from the key GCObject.
+ if /* constexpr */ (sizeof(uintptr_t) > 4) {
+ LUAU_ASSERT(sizeof(uintptr_t) <= 8);
+ gval(n)->tt = static_cast(static_cast(static_cast(intptr) >> 32));
+ }
+ return n;
+}
+
+/**
+ * @brief Relink the ephemeron list of a key to the worklist of the current
+ * ephemeron key marking to ensure that no recursion is occuring.
+ *
+ * @param o Ephemeron key, used to restore the key.
+ * @param first Pointer to first element, will be restored to the original value.
+ * @param list Worklist
+ * @return LuaNode* Adapted worklist
+ */
+static LuaNode* preparelist(GCObject* o, void** first, LuaNode* list) {
+ void* c = *first;
+ LuaNode* n;
+ int last;
+ do {
+ n = reinterpret_cast(c); // c points to a LuaNode
+ last = gkey(n)->extra[0] & kLastElement; // Is last element?
+ c = gkey(n)->value.p; // Pointer to next ephemeron list element
+ list = relink(o, n, list);
+ } while(!last);
+ // Restore original value, since in userdata we use the metatable ptr for the list ptr.
+ *first = c == nullptr ? nullptr : reinterpret_cast(c) - 1;
+ return list;
+}
+
+/**
+ * @brief Restore the table node from the worklist.
+ *
+ * @param n Node to restore to the original state
+ * @return LuaNode* Next node on the worklist
+ */
+static LuaNode* nextandrestore(LuaNode* n) {
+ uintptr_t intptr = static_cast(gkey(n)->extra[0]);
+ // Rebuild ptr to next element in the worklist which is stored in the key extra value
+ // and the tt slot of the value.
+ if /* constexpr */ (sizeof(uintptr_t) > 4) {
+ intptr |= static_cast(static_cast(static_cast(gval(n)->tt)) << 32);
+ }
+ ttype(gval(n)) = ttype(gkey(n)); // Value tt is saved in the key tt.
+ ttype(gkey(n)) = gcvalue(gkey(n))->gch.tt; // Key tt can be restored from the key gc value.
+ return reinterpret_cast(intptr);
+}
+
+/**
+ * @brief Mark object o. Special case userdata, since the metatable can be an ephemeron key which would cause recursion.
+ *
+ * @param g Global Lua state
+ * @param o Object to mark
+ * @param n Worklist
+ * @return LuaNode* Adapted worklist
+ */
+static LuaNode* reallymarkobjecthandleephemeron(global_State* g, GCObject* o, LuaNode* n) {
+ if (o->gch.tt == LUA_TUSERDATA) {
+ // Special case userdata to avoid recursion in case metatable is an ephemeron key.
+ Table* mt = gco2u(o)->metatable;
+ white2gray(o);
+ gray2black(o);
+ if (mt) {
+ o = obj2gco(mt);
+ if (isephemeronkey(o)) {
+ n = preparelist(o, getephemeronlink(o), n);
+ mt->marked &= ~bitmask(EPHEMERONKEYBIT);
+ }
+ reallymarkobject0(g, o); // Note: o is a table which will be linked to gray list, so no recursion can occure.
+ }
+ } else {
+ // All other objects will be linked to the gray list.
+ reallymarkobject0(g, o);
+ }
+ return n;
+}
+
+/**
+ * @brief Marks the ephemeron key o. This restores all the elements of the ephemeron link and marks all values.
+ * This is done with a working list as values might be itself ephemeron keys and recursion might result in stack overflows.
+ *
+ * @param g Global Lua state
+ * @param o Ephemeron key to mark
+ */
+static void markephemeronkey(global_State* g, GCObject* o) {
+ // Move the ephemeron list to the worklist
+ LuaNode* list = preparelist(o, getephemeronlink(o), nullptr);
+ o->gch.marked &= ~bitmask(EPHEMERONKEYBIT);
+ list = reallymarkobjecthandleephemeron(g, o, list); // Mark the object.
+ while (list) {
+ // Iterate through the worklist
+ LuaNode* next = nextandrestore(list);
+ if (iscollectable(gval(list))) {
+ o = gcvalue(gval(list));
+ if (iswhite(o)) {
+ // Value is collectable and needs marking
+ if (isephemeronkey(o)) {
+ // Special case ephemeron keys
+ next = preparelist(o, getephemeronlink(o), next);
+ o->gch.marked &= ~bitmask(EPHEMERONKEYBIT);
+ }
+ next = reallymarkobjecthandleephemeron(g, o, next); // Mark the object.
+ }
+ }
+ list = next;
+ }
+}
+
+template
+static void reallymarkobject(global_State* g, GCObject* o)
+{
+ LUAU_ASSERT(iswhite(o) && !isdead(g, o));
+ if (A && isephemeronkey(o)) {
+ // Check if the object is an ephemeron key.
+ // This should only be done and the case in the atomic phase.
+ markephemeronkey(g, o);
+ } else {
+ reallymarkobject0(g, o);
+ }
+}
+
static const char* gettablemode(global_State* g, Table* h)
{
const TValue* mode = gfasttm(g, h->metatable, TM_MODE);
@@ -193,33 +364,125 @@ static const char* gettablemode(global_State* g, Table* h)
return NULL;
}
+/**
+ * @brief Make the key of table node n an ephemeron key if it is not already and link the table node n to the list.
+ * While the node is in the ephemeron list the node will look like a dead node.
+ * The key will be marked as LUA_TDEADKEY and the value as LUA_TNIL.
+ * This makes the case were the key is actually dead very cheap.
+ *
+ * @param g Global Lua state
+ * @param n Table node to link to the nodes key ephemeron list.
+ */
+static void linkephemeronkey(global_State* g, LuaNode* n) {
+ GCObject* o = gcvalue(gkey(n));
+ void** link = getephemeronlink(o);
+ void* ptr;
+ int extra = ttype(gval(n));
+ if (isephemeronkey(o)) {
+ ptr = *link;
+ } else {
+ l_setbit(o->gch.marked, EPHEMERONKEYBIT);
+ if (o->gch.tt == LUA_TUSERDATA) {
+ // We need to increate the pointer since the metatable might be as key in this table and when it is marked
+ // as LUA_TDEADKEY has the same pointer as in this node. This might result in problems, so ensure the pointer
+ // is not a pointer to a valid Lua GCObject.
+ ptr = reinterpret_cast(gco2u(o)->metatable) + 1;
+ } else {
+ ptr = nullptr;
+ }
+ extra |= kLastElement;
+ }
+
+ gkey(n)->value.p = ptr; // Pointer to next element in the ephemeron list.
+ gkey(n)->extra[0] = extra; // Save tt of the value.
+ ttype(gkey(n)) = LUA_TDEADKEY; // Make this node a dead node.
+ ttype(gval(n)) = LUA_TNIL;
+ *link = n;
+}
+
+template
+static int traverseephemeron(global_State* g, Table* h, const char* modev) {
+ int i;
+ int hasweakkey = 0;
+ i = h->sizearray;
+ while (i--)
+ markvalue(A, g, &h->array[i]);
+ i = sizenode(h);
+ while (i--)
+ {
+ LuaNode* n = gnode(h, i);
+ LUAU_ASSERT(ttype(gkey(n)) != LUA_TDEADKEY || ttisnil(gval(n)));
+ if (ttisnil(gval(n)))
+ removeentry(n); // remove empty entries
+ else
+ {
+ LUAU_ASSERT(!ttisnil(gkey(n)));
+ if (iscollectable(gkey(n))) {
+ if (ttype(gkey(n)) == LUA_TSTRING) {
+ markvalue(A, g, gkey(n));
+ markvalue(A, g, gval(n));
+ } else if (!iswhite(gcvalue(gkey(n)))){
+ markvalue(A, g, gval(n));
+ } else if /* constexpr */ (A) {
+ linkephemeronkey(g, n);
+ } else {
+ hasweakkey = 1;
+ }
+ } else {
+ markvalue(A, g, gval(n));
+ }
+ }
+ }
+ if (hasweakkey) {
+ // Non trivial case in the propagating/marking phase.
+ // So ensure this object is visited again in the atomic phase.
+ h->gclist = g->grayagain;
+ g->grayagain = obj2gco(h);
+ } else if (strchr(modev, 's') != NULL) {
+ // Shrinkable ephemeron table.
+ h->gclist = g->weak; // link to weak list for shrinking.
+ g->weak = obj2gco(h);
+ return 1;
+ }
+ return hasweakkey;
+}
+
+template
static int traversetable(global_State* g, Table* h)
{
int i;
int weakkey = 0;
int weakvalue = 0;
if (h->metatable)
- markobject(g, cast_to(Table*, h->metatable));
+ markobject(A, g, cast_to(Table*, h->metatable));
// is there a weak mode?
if (const char* modev = gettablemode(g, h))
{
weakkey = (strchr(modev, 'k') != NULL);
weakvalue = (strchr(modev, 'v') != NULL);
- if (weakkey || weakvalue)
+ if (weakkey)
+ {
+ if (weakvalue)
+ { // is really weak?
+ h->gclist = g->weak; // must be cleared after GC, ...
+ g->weak = obj2gco(h); // ... so put in the appropriate list
+ return 1;
+ }
+ return traverseephemeron(g, h, modev);
+ }
+ if (weakvalue)
{ // is really weak?
h->gclist = g->weak; // must be cleared after GC, ...
g->weak = obj2gco(h); // ... so put in the appropriate list
}
}
- if (weakkey && weakvalue)
- return 1;
if (!weakvalue)
{
i = h->sizearray;
while (i--)
- markvalue(g, &h->array[i]);
+ markvalue(A, g, &h->array[i]);
}
i = sizenode(h);
while (i--)
@@ -232,9 +495,9 @@ static int traversetable(global_State* g, Table* h)
{
LUAU_ASSERT(!ttisnil(gkey(n)));
if (!weakkey)
- markvalue(g, gkey(n));
+ markvalue(A, g, gkey(n));
if (!weakvalue)
- markvalue(g, gval(n));
+ markvalue(A, g, gval(n));
}
}
return weakkey || weakvalue;
@@ -244,6 +507,7 @@ static int traversetable(global_State* g, Table* h)
** All marks are conditional because a GC may happen while the
** prototype is still being created
*/
+template
static void traverseproto(global_State* g, Proto* f)
{
int i;
@@ -252,7 +516,7 @@ static void traverseproto(global_State* g, Proto* f)
if (f->debugname)
stringmark(f->debugname);
for (i = 0; i < f->sizek; i++) // mark literals
- markvalue(g, &f->k[i]);
+ markvalue(A, g, &f->k[i]);
for (i = 0; i < f->sizeupvalues; i++)
{ // mark upvalue names
if (f->upvalues[i])
@@ -261,7 +525,7 @@ static void traverseproto(global_State* g, Proto* f)
for (i = 0; i < f->sizep; i++)
{ // mark nested protos
if (f->p[i])
- markobject(g, f->p[i]);
+ markobject(A, g, f->p[i]);
}
for (i = 0; i < f->sizelocvars; i++)
{ // mark local-variable names
@@ -270,32 +534,34 @@ static void traverseproto(global_State* g, Proto* f)
}
}
+template
static void traverseclosure(global_State* g, Closure* cl)
{
- markobject(g, cl->env);
+ markobject(A, g, cl->env);
if (cl->isC)
{
int i;
for (i = 0; i < cl->nupvalues; i++) // mark its upvalues
- markvalue(g, &cl->c.upvals[i]);
+ markvalue(A, g, &cl->c.upvals[i]);
}
else
{
int i;
LUAU_ASSERT(cl->nupvalues == cl->l.p->nups);
- markobject(g, cast_to(Proto*, cl->l.p));
+ markobject(A, g, cast_to(Proto*, cl->l.p));
for (i = 0; i < cl->nupvalues; i++) // mark its upvalues
- markvalue(g, &cl->l.uprefs[i]);
+ markvalue(A, g, &cl->l.uprefs[i]);
}
}
+template
static void traversestack(global_State* g, lua_State* l, bool clearstack)
{
- markobject(g, l->gt);
+ markobject(A, g, l->gt);
if (l->namecall)
stringmark(l->namecall);
for (StkId o = l->stack; o < l->top; o++)
- markvalue(g, o);
+ markvalue(A, g, o);
// final traversal?
if (g->gcstate == GCSatomic || clearstack)
{
@@ -309,6 +575,7 @@ static void traversestack(global_State* g, lua_State* l, bool clearstack)
** traverse one gray object, turning it to black.
** Returns `quantity' traversed.
*/
+template
static size_t propagatemark(global_State* g)
{
GCObject* o = g->gray;
@@ -320,7 +587,7 @@ static size_t propagatemark(global_State* g)
{
Table* h = gco2h(o);
g->gray = h->gclist;
- if (traversetable(g, h)) // table is weak?
+ if (traversetable(g, h)) // table is weak?
black2gray(o); // keep it gray
return sizeof(Table) + sizeof(TValue) * h->sizearray + sizeof(LuaNode) * sizenode(h);
}
@@ -328,7 +595,7 @@ static size_t propagatemark(global_State* g)
{
Closure* cl = gco2cl(o);
g->gray = cl->gclist;
- traverseclosure(g, cl);
+ traverseclosure(g, cl);
return cl->isC ? sizeCclosure(cl->nupvalues) : sizeLclosure(cl->nupvalues);
}
case LUA_TTHREAD:
@@ -343,7 +610,7 @@ static size_t propagatemark(global_State* g)
if (!active && g->gcstate == GCSpropagate)
{
- traversestack(g, th, /* clearstack= */ true);
+ traversestack(g, th, /* clearstack= */ true);
l_setbit(th->stackstate, THREAD_SLEEPINGBIT);
}
@@ -354,7 +621,7 @@ static size_t propagatemark(global_State* g)
black2gray(o);
- traversestack(g, th, /* clearstack= */ false);
+ traversestack(g, th, /* clearstack= */ false);
}
return sizeof(lua_State) + sizeof(TValue) * th->stacksize + sizeof(CallInfo) * th->size_ci;
@@ -363,7 +630,7 @@ static size_t propagatemark(global_State* g)
{
Proto* p = gco2p(o);
g->gray = p->gclist;
- traverseproto(g, p);
+ traverseproto(g, p);
return sizeof(Proto) + sizeof(Instruction) * p->sizecode + sizeof(Proto*) * p->sizep + sizeof(TValue) * p->sizek + p->sizelineinfo +
sizeof(LocVar) * p->sizelocvars + sizeof(TString*) * p->sizeupvalues;
}
@@ -373,12 +640,13 @@ static size_t propagatemark(global_State* g)
}
}
+template
static size_t propagateall(global_State* g)
{
size_t work = 0;
while (g->gray)
{
- work += propagatemark(g);
+ work += propagatemark(g);
}
return work;
}
@@ -569,29 +837,32 @@ void luaC_freeall(lua_State* L)
LUAU_ASSERT(g->strbufgc == NULL);
}
+template
static void markmt(global_State* g)
{
int i;
for (i = 0; i < LUA_T_COUNT; i++)
if (g->mt[i])
- markobject(g, g->mt[i]);
+ markobject(A, g, g->mt[i]);
}
// mark root set
+template
static void markroot(lua_State* L)
{
global_State* g = L->global;
g->gray = NULL;
g->grayagain = NULL;
g->weak = NULL;
- markobject(g, g->mainthread);
+ markobject(A, g, g->mainthread);
// make global table be traversed before main stack
- markobject(g, g->mainthread->gt);
- markvalue(g, registry(L));
- markmt(g);
+ markobject(A, g, g->mainthread->gt);
+ markvalue(A, g, registry(L));
+ markmt(g);
g->gcstate = GCSpropagate;
}
+template
static size_t remarkupvals(global_State* g)
{
size_t work = 0;
@@ -600,7 +871,7 @@ static size_t remarkupvals(global_State* g)
work += sizeof(UpVal);
LUAU_ASSERT(uv->u.l.next->u.l.prev == uv && uv->u.l.prev->u.l.next == uv);
if (isgray(obj2gco(uv)))
- markvalue(g, uv->v);
+ markvalue(A, g, uv->v);
}
return work;
}
@@ -617,9 +888,9 @@ static size_t atomic(lua_State* L)
#endif
// remark occasional upvalues of (maybe) dead threads
- work += remarkupvals(g);
+ work += remarkupvals(g);
// traverse objects caught by write barrier and by 'remarkupvals'
- work += propagateall(g);
+ work += propagateall(g);
#ifdef LUAI_GCMETRICS
g->gcmetrics.currcycle.atomictimeupval += recordGcDeltaTime(currts);
@@ -629,9 +900,9 @@ static size_t atomic(lua_State* L)
g->gray = g->weak;
g->weak = NULL;
LUAU_ASSERT(!iswhite(obj2gco(g->mainthread)));
- markobject(g, L); // mark running thread
- markmt(g); // mark basic metatables (again)
- work += propagateall(g);
+ markobject(true, g, L); // mark running thread
+ markmt(g); // mark basic metatables (again)
+ work += propagateall(g);
#ifdef LUAI_GCMETRICS
g->gcmetrics.currcycle.atomictimeweak += recordGcDeltaTime(currts);
@@ -640,7 +911,7 @@ static size_t atomic(lua_State* L)
// remark gray again
g->gray = g->grayagain;
g->grayagain = NULL;
- work += propagateall(g);
+ work += propagateall(g);
#ifdef LUAI_GCMETRICS
g->gcmetrics.currcycle.atomictimegray += recordGcDeltaTime(currts);
@@ -733,7 +1004,7 @@ static size_t gcstep(lua_State* L, size_t limit)
{
case GCSpause:
{
- markroot(L); // start a new collection
+ markroot(L); // start a new collection
LUAU_ASSERT(g->gcstate == GCSpropagate);
break;
}
@@ -741,7 +1012,7 @@ static size_t gcstep(lua_State* L, size_t limit)
{
while (g->gray && cost < limit)
{
- cost += propagatemark(g);
+ cost += propagatemark(g);
}
if (!g->gray)
@@ -762,7 +1033,7 @@ static size_t gcstep(lua_State* L, size_t limit)
{
while (g->gray && cost < limit)
{
- cost += propagatemark(g);
+ cost += propagatemark(g);
}
if (!g->gray) // no more `gray' objects
@@ -969,7 +1240,7 @@ void luaC_fullgc(lua_State* L)
#endif
// run a full collection cycle
- markroot(L);
+ markroot(L);
while (g->gcstate != GCSpause)
{
gcstep(L, SIZE_MAX);
@@ -1003,7 +1274,7 @@ void luaC_barrierupval(lua_State* L, GCObject* v)
LUAU_ASSERT(iswhite(v) && !isdead(g, v));
if (keepinvariant(g))
- reallymarkobject(g, v);
+ reallymarkobject(g, v);
}
void luaC_barrierf(lua_State* L, GCObject* o, GCObject* v)
@@ -1013,7 +1284,7 @@ void luaC_barrierf(lua_State* L, GCObject* o, GCObject* v)
LUAU_ASSERT(g->gcstate != GCSpause);
// must keep invariant?
if (keepinvariant(g))
- reallymarkobject(g, v); // restore invariant
+ reallymarkobject(g, v); // restore invariant
else // don't mind
makewhite(g, o); // mark as white just to avoid other barriers
}
@@ -1027,7 +1298,7 @@ void luaC_barriertable(lua_State* L, Table* t, GCObject* v)
if (g->gcstate == GCSpropagateagain)
{
LUAU_ASSERT(isblack(o) && iswhite(v) && !isdead(g, v) && !isdead(g, o));
- reallymarkobject(g, v);
+ reallymarkobject(g, v);
return;
}
diff --git a/VM/src/lgc.h b/VM/src/lgc.h
index 7b03a25d6..833bf7e32 100644
--- a/VM/src/lgc.h
+++ b/VM/src/lgc.h
@@ -52,18 +52,21 @@
** bit 1 - object is white (type 1)
** bit 2 - object is black
** bit 3 - object is fixed (should not be collected)
+** bit 4 - object is an ephemeron key and needs special handling when marked from white to grey or black (only used in the atomic gc phase)
*/
#define WHITE0BIT 0
#define WHITE1BIT 1
#define BLACKBIT 2
#define FIXEDBIT 3
+#define EPHEMERONKEYBIT 4
#define WHITEBITS bit2mask(WHITE0BIT, WHITE1BIT)
#define iswhite(x) test2bits((x)->gch.marked, WHITE0BIT, WHITE1BIT)
#define isblack(x) testbit((x)->gch.marked, BLACKBIT)
#define isgray(x) (!testbits((x)->gch.marked, WHITEBITS | bitmask(BLACKBIT)))
#define isfixed(x) testbit((x)->gch.marked, FIXEDBIT)
+#define isephemeronkey(x) testbit((x)->gch.marked, EPHEMERONKEYBIT)
#define otherwhite(g) (g->currentwhite ^ WHITEBITS)
#define isdead(g, v) (((v)->gch.marked & (WHITEBITS | bitmask(FIXEDBIT))) == (otherwhite(g) & WHITEBITS))
diff --git a/tests/conformance/gc.lua b/tests/conformance/gc.lua
index 5804ea7f7..902296a50 100644
--- a/tests/conformance/gc.lua
+++ b/tests/conformance/gc.lua
@@ -132,14 +132,14 @@ print('weak tables')
a = {}; setmetatable(a, {__mode = 'k'});
-- fill a with some `collectable' indices
for i=1,lim do a[{}] = i end
--- and some non-collectable ones
for i=1,lim do local t={}; a[t]=t end
+-- and some non-collectable ones
for i=1,lim do a[i] = i end
for i=1,lim do local s=string.rep('@', i); a[s] = s..'#' end
collectgarbage()
local i = 0
for k,v in pairs(a) do assert(k==v or k..'#'==v); i=i+1 end
-assert(i == 3*lim)
+assert(i == 2*lim)
a = {}; setmetatable(a, {__mode = 'v'});
a[1] = string.rep('b', 21)