// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details // This code is based on Lua 5.x implementation licensed under MIT License; see lua_LICENSE.txt for details #include "lgc.h" #include "lobject.h" #include "lstate.h" #include "ltable.h" #include "lfunc.h" #include "lstring.h" #include "ldo.h" #include #include LUAU_FASTFLAGVARIABLE(LuauRescanGrayAgain, false) LUAU_FASTFLAGVARIABLE(LuauRescanGrayAgainForwardBarrier, false) LUAU_FASTFLAGVARIABLE(LuauGcFullSkipInactiveThreads, false) LUAU_FASTFLAGVARIABLE(LuauShrinkWeakTables, false) LUAU_FASTFLAG(LuauArrayBoundary) #define GC_SWEEPMAX 40 #define GC_SWEEPCOST 10 #define GC_INTERRUPT(state) \ { \ void (*interrupt)(lua_State*, int) = g->cb.interrupt; \ if (LUAU_UNLIKELY(!!interrupt)) \ interrupt(L, state); \ } #define maskmarks cast_byte(~(bitmask(BLACKBIT) | WHITEBITS)) #define makewhite(g, x) ((x)->gch.marked = cast_byte(((x)->gch.marked & maskmarks) | luaC_white(g))) #define white2gray(x) reset2bits((x)->gch.marked, WHITE0BIT, WHITE1BIT) #define black2gray(x) resetbit((x)->gch.marked, BLACKBIT) #define stringmark(s) reset2bits((s)->marked, WHITE0BIT, WHITE1BIT) #define markvalue(g, o) \ { \ checkconsistency(o); \ if (iscollectable(o) && iswhite(gcvalue(o))) \ reallymarkobject(g, gcvalue(o)); \ } #define markobject(g, t) \ { \ if (iswhite(obj2gco(t))) \ reallymarkobject(g, obj2gco(t)); \ } static void recordGcStateTime(global_State* g, int startgcstate, double seconds, bool assist) { switch (startgcstate) { case GCSpause: // record root mark time if we have switched to next state if (g->gcstate == GCSpropagate) g->gcstats.currcycle.marktime += seconds; break; case GCSpropagate: case GCSpropagateagain: g->gcstats.currcycle.marktime += seconds; // atomic step had to be performed during the switch and it's tracked separately if (g->gcstate == GCSsweepstring) g->gcstats.currcycle.marktime -= g->gcstats.currcycle.atomictime; break; case GCSsweepstring: case GCSsweep: g->gcstats.currcycle.sweeptime += seconds; break; } if (assist) g->gcstats.stepassisttimeacc += seconds; else g->gcstats.stepexplicittimeacc += seconds; } static void startGcCycleStats(global_State* g) { g->gcstats.currcycle.starttimestamp = lua_clock(); g->gcstats.currcycle.waittime = g->gcstats.currcycle.starttimestamp - g->gcstats.lastcycle.endtimestamp; } static void finishGcCycleStats(global_State* g) { g->gcstats.currcycle.endtimestamp = lua_clock(); g->gcstats.currcycle.endtotalsizebytes = g->totalbytes; g->gcstats.completedcycles++; g->gcstats.lastcycle = g->gcstats.currcycle; g->gcstats.currcycle = GCCycleStats(); g->gcstats.cyclestatsacc.markitems += g->gcstats.lastcycle.markitems; g->gcstats.cyclestatsacc.marktime += g->gcstats.lastcycle.marktime; g->gcstats.cyclestatsacc.atomictime += g->gcstats.lastcycle.atomictime; g->gcstats.cyclestatsacc.sweepitems += g->gcstats.lastcycle.sweepitems; g->gcstats.cyclestatsacc.sweeptime += g->gcstats.lastcycle.sweeptime; } static void removeentry(LuaNode* n) { LUAU_ASSERT(ttisnil(gval(n))); if (iscollectable(gkey(n))) setttype(gkey(n), LUA_TDEADKEY); /* dead key; remove it */ } static void reallymarkobject(global_State* g, GCObject* o) { LUAU_ASSERT(iswhite(o) && !isdead(g, o)); white2gray(o); switch (o->gch.tt) { case LUA_TSTRING: { return; } case LUA_TUSERDATA: { Table* mt = gco2u(o)->metatable; gray2black(o); /* udata are never gray */ if (mt) markobject(g, mt); return; } case LUA_TUPVAL: { UpVal* uv = gco2uv(o); markvalue(g, uv->v); if (uv->v == &uv->u.value) /* closed? */ gray2black(o); /* open upvalues are never black */ return; } case LUA_TFUNCTION: { gco2cl(o)->gclist = g->gray; g->gray = o; break; } case LUA_TTABLE: { gco2h(o)->gclist = g->gray; g->gray = o; break; } case LUA_TTHREAD: { gco2th(o)->gclist = g->gray; g->gray = o; break; } case LUA_TPROTO: { gco2p(o)->gclist = g->gray; g->gray = o; break; } default: LUAU_ASSERT(0); } } static const char* gettablemode(global_State* g, Table* h) { const TValue* mode = gfasttm(g, h->metatable, TM_MODE); if (mode && ttisstring(mode)) return svalue(mode); return NULL; } 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)); if (FFlag::LuauShrinkWeakTables) { /* 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) { /* is really weak? */ h->gclist = g->weak; /* must be cleared after GC, ... */ g->weak = obj2gco(h); /* ... so put in the appropriate list */ } } } else { const TValue* mode = gfasttm(g, h->metatable, TM_MODE); if (mode && ttisstring(mode)) { /* is there a weak mode? */ const char* modev = svalue(mode); weakkey = (strchr(modev, 'k') != NULL); weakvalue = (strchr(modev, 'v') != NULL); if (weakkey || 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]); } 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 (!weakkey) markvalue(g, gkey(n)); if (!weakvalue) markvalue(g, gval(n)); } } return weakkey || weakvalue; } /* ** All marks are conditional because a GC may happen while the ** prototype is still being created */ static void traverseproto(global_State* g, Proto* f) { int i; if (f->source) stringmark(f->source); if (f->debugname) stringmark(f->debugname); for (i = 0; i < f->sizek; i++) /* mark literals */ markvalue(g, &f->k[i]); for (i = 0; i < f->sizeupvalues; i++) { /* mark upvalue names */ if (f->upvalues[i]) stringmark(f->upvalues[i]); } for (i = 0; i < f->sizep; i++) { /* mark nested protos */ if (f->p[i]) markobject(g, f->p[i]); } for (i = 0; i < f->sizelocvars; i++) { /* mark local-variable names */ if (f->locvars[i].varname) stringmark(f->locvars[i].varname); } } static void traverseclosure(global_State* g, Closure* cl) { markobject(g, cl->env); if (cl->isC) { int i; for (i = 0; i < cl->nupvalues; i++) /* mark its upvalues */ markvalue(g, &cl->c.upvals[i]); } else { int i; LUAU_ASSERT(cl->nupvalues == cl->l.p->nups); markobject(g, cast_to(Proto*, cl->l.p)); for (i = 0; i < cl->nupvalues; i++) /* mark its upvalues */ markvalue(g, &cl->l.uprefs[i]); } } static void traversestack(global_State* g, lua_State* l, bool clearstack) { markvalue(g, gt(l)); if (l->namecall) stringmark(l->namecall); for (StkId o = l->stack; o < l->top; o++) markvalue(g, o); /* final traversal? */ if (g->gcstate == GCSatomic || (FFlag::LuauGcFullSkipInactiveThreads && clearstack)) { StkId stack_end = l->stack + l->stacksize; for (StkId o = l->top; o < stack_end; o++) /* clear not-marked stack slice */ setnilvalue(o); } } /* ** traverse one gray object, turning it to black. ** Returns `quantity' traversed. */ static size_t propagatemark(global_State* g) { GCObject* o = g->gray; LUAU_ASSERT(isgray(o)); gray2black(o); switch (o->gch.tt) { case LUA_TTABLE: { Table* h = gco2h(o); g->gray = h->gclist; if (traversetable(g, h)) /* table is weak? */ black2gray(o); /* keep it gray */ return sizeof(Table) + sizeof(TValue) * h->sizearray + sizeof(LuaNode) * sizenode(h); } case LUA_TFUNCTION: { Closure* cl = gco2cl(o); g->gray = cl->gclist; traverseclosure(g, cl); return cl->isC ? sizeCclosure(cl->nupvalues) : sizeLclosure(cl->nupvalues); } case LUA_TTHREAD: { lua_State* th = gco2th(o); g->gray = th->gclist; if (FFlag::LuauGcFullSkipInactiveThreads) { LUAU_ASSERT(!luaC_threadsleeping(th)); // threads that are executing and the main thread are not deactivated bool active = luaC_threadactive(th) || th == th->global->mainthread; if (!active && g->gcstate == GCSpropagate) { traversestack(g, th, /* clearstack= */ true); l_setbit(th->stackstate, THREAD_SLEEPINGBIT); } else { th->gclist = g->grayagain; g->grayagain = o; black2gray(o); traversestack(g, th, /* clearstack= */ false); } } else { th->gclist = g->grayagain; g->grayagain = o; black2gray(o); traversestack(g, th, /* clearstack= */ false); } return sizeof(lua_State) + sizeof(TValue) * th->stacksize + sizeof(CallInfo) * th->size_ci; } case LUA_TPROTO: { Proto* p = gco2p(o); g->gray = p->gclist; 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; } default: LUAU_ASSERT(0); return 0; } } static void propagateall(global_State* g) { while (g->gray) { propagatemark(g); } } /* ** The next function tells whether a key or value can be cleared from ** a weak table. Non-collectable objects are never removed from weak ** tables. Strings behave as `values', so are never removed too. for ** other objects: if really collected, cannot keep them. */ static int isobjcleared(GCObject* o) { if (o->gch.tt == LUA_TSTRING) { stringmark(&o->ts); /* strings are `values', so are never weak */ return 0; } return iswhite(o); } #define iscleared(o) (iscollectable(o) && isobjcleared(gcvalue(o))) /* ** clear collected entries from weaktables */ static void cleartable(lua_State* L, GCObject* l) { while (l) { Table* h = gco2h(l); int i = h->sizearray; while (i--) { TValue* o = &h->array[i]; if (iscleared(o)) /* value was collected? */ setnilvalue(o); /* remove value */ } i = sizenode(h); int activevalues = 0; while (i--) { LuaNode* n = gnode(h, i); if (FFlag::LuauShrinkWeakTables) { // non-empty entry? if (!ttisnil(gval(n))) { // can we clear key or value? if (iscleared(gkey(n)) || iscleared(gval(n))) { setnilvalue(gval(n)); /* remove value ... */ removeentry(n); /* remove entry from table */ } else { activevalues++; } } } else { if (!ttisnil(gval(n)) && /* non-empty entry? */ (iscleared(gkey(n)) || iscleared(gval(n)))) { setnilvalue(gval(n)); /* remove value ... */ removeentry(n); /* remove entry from table */ } } } if (FFlag::LuauShrinkWeakTables) { if (const char* modev = gettablemode(L->global, h)) { // are we allowed to shrink this weak table? if (strchr(modev, 's')) { // shrink at 37.5% occupancy if (activevalues < sizenode(h) * 3 / 8) luaH_resizehash(L, h, activevalues); } } } l = h->gclist; } } static void shrinkstack(lua_State* L) { /* compute used stack - note that we can't use th->top if we're in the middle of vararg call */ StkId lim = L->top; for (CallInfo* ci = L->base_ci; ci <= L->ci; ci++) { LUAU_ASSERT(ci->top <= L->stack_last); if (lim < ci->top) lim = ci->top; } /* shrink stack and callinfo arrays if we aren't using most of the space */ int ci_used = cast_int(L->ci - L->base_ci); /* number of `ci' in use */ int s_used = cast_int(lim - L->stack); /* part of stack in use */ if (L->size_ci > LUAI_MAXCALLS) /* handling overflow? */ return; /* do not touch the stacks */ if (3 * ci_used < L->size_ci && 2 * BASIC_CI_SIZE < L->size_ci) luaD_reallocCI(L, L->size_ci / 2); /* still big enough... */ condhardstacktests(luaD_reallocCI(L, ci_used + 1)); if (3 * s_used < L->stacksize && 2 * (BASIC_STACK_SIZE + EXTRA_STACK) < L->stacksize) luaD_reallocstack(L, L->stacksize / 2); /* still big enough... */ condhardstacktests(luaD_reallocstack(L, s_used)); } static void freeobj(lua_State* L, GCObject* o) { switch (o->gch.tt) { case LUA_TPROTO: luaF_freeproto(L, gco2p(o)); break; case LUA_TFUNCTION: luaF_freeclosure(L, gco2cl(o)); break; case LUA_TUPVAL: luaF_freeupval(L, gco2uv(o)); break; case LUA_TTABLE: luaH_free(L, gco2h(o)); break; case LUA_TTHREAD: LUAU_ASSERT(gco2th(o) != L && gco2th(o) != L->global->mainthread); luaE_freethread(L, gco2th(o)); break; case LUA_TSTRING: luaS_free(L, gco2ts(o)); break; case LUA_TUSERDATA: luaS_freeudata(L, gco2u(o)); break; default: LUAU_ASSERT(0); } } #define sweepwholelist(L, p, tc) sweeplist(L, p, SIZE_MAX, tc) static GCObject** sweeplist(lua_State* L, GCObject** p, size_t count, size_t* traversedcount) { GCObject* curr; global_State* g = L->global; int deadmask = otherwhite(g); size_t startcount = count; LUAU_ASSERT(testbit(deadmask, FIXEDBIT)); /* make sure we never sweep fixed objects */ while ((curr = *p) != NULL && count-- > 0) { int alive = (curr->gch.marked ^ WHITEBITS) & deadmask; if (curr->gch.tt == LUA_TTHREAD) { sweepwholelist(L, &gco2th(curr)->openupval, traversedcount); /* sweep open upvalues */ lua_State* th = gco2th(curr); if (alive) { resetbit(th->stackstate, THREAD_SLEEPINGBIT); shrinkstack(th); } } if (alive) { /* not dead? */ LUAU_ASSERT(!isdead(g, curr)); makewhite(g, curr); /* make it white (for next cycle) */ p = &curr->gch.next; } else { /* must erase `curr' */ LUAU_ASSERT(isdead(g, curr)); *p = curr->gch.next; if (curr == g->rootgc) /* is the first element of the list? */ g->rootgc = curr->gch.next; /* adjust first */ freeobj(L, curr); } } // if we didn't reach the end of the list it means that we've stopped because the count dropped below zero if (traversedcount) *traversedcount += startcount - (curr ? count + 1 : count); return p; } static void deletelist(lua_State* L, GCObject** p, GCObject* limit) { GCObject* curr; while ((curr = *p) != limit) { if (curr->gch.tt == LUA_TTHREAD) /* delete open upvalues of each thread */ deletelist(L, &gco2th(curr)->openupval, NULL); *p = curr->gch.next; freeobj(L, curr); } } static void shrinkbuffers(lua_State* L) { global_State* g = L->global; /* check size of string hash */ if (g->strt.nuse < cast_to(uint32_t, g->strt.size / 4) && g->strt.size > LUA_MINSTRTABSIZE * 2) luaS_resize(L, g->strt.size / 2); /* table is too big */ } static void shrinkbuffersfull(lua_State* L) { global_State* g = L->global; /* check size of string hash */ int hashsize = g->strt.size; while (g->strt.nuse < cast_to(uint32_t, hashsize / 4) && hashsize > LUA_MINSTRTABSIZE * 2) hashsize /= 2; if (hashsize != g->strt.size) luaS_resize(L, hashsize); /* table is too big */ } void luaC_freeall(lua_State* L) { global_State* g = L->global; LUAU_ASSERT(L == g->mainthread); LUAU_ASSERT(L->next == NULL); /* mainthread is at the end of rootgc list */ deletelist(L, &g->rootgc, obj2gco(L)); for (int i = 0; i < g->strt.size; i++) /* free all string lists */ deletelist(L, &g->strt.hash[i], NULL); LUAU_ASSERT(L->global->strt.nuse == 0); deletelist(L, &g->strbufgc, NULL); // unfortunately, when string objects are freed, the string table use count is decremented // even when the string is a buffer that wasn't placed into the table L->global->strt.nuse = 0; } 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]); } /* mark root set */ static void markroot(lua_State* L) { global_State* g = L->global; g->gray = NULL; g->grayagain = NULL; g->weak = NULL; markobject(g, g->mainthread); /* make global table be traversed before main stack */ markvalue(g, gt(g->mainthread)); markvalue(g, registry(L)); markmt(g); g->gcstate = GCSpropagate; } static void remarkupvals(global_State* g) { UpVal* uv; for (uv = g->uvhead.u.l.next; uv != &g->uvhead; uv = uv->u.l.next) { 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); } } static void atomic(lua_State* L) { global_State* g = L->global; g->gcstate = GCSatomic; /* remark occasional upvalues of (maybe) dead threads */ remarkupvals(g); /* traverse objects caught by write barrier and by 'remarkupvals' */ propagateall(g); /* remark weak tables */ 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) */ propagateall(g); /* remark gray again */ g->gray = g->grayagain; g->grayagain = NULL; propagateall(g); cleartable(L, g->weak); /* remove collected objects from weak tables */ g->weak = NULL; /* flip current white */ g->currentwhite = cast_byte(otherwhite(g)); g->sweepstrgc = 0; g->sweepgc = &g->rootgc; g->gcstate = GCSsweepstring; GC_INTERRUPT(GCSatomic); } static size_t singlestep(lua_State* L) { size_t cost = 0; global_State* g = L->global; switch (g->gcstate) { case GCSpause: { markroot(L); /* start a new collection */ break; } case GCSpropagate: { if (FFlag::LuauRescanGrayAgain) { if (g->gray) { g->gcstats.currcycle.markitems++; cost = propagatemark(g); } else { // perform one iteration over 'gray again' list g->gray = g->grayagain; g->grayagain = NULL; g->gcstate = GCSpropagateagain; } } else { if (g->gray) { g->gcstats.currcycle.markitems++; cost = propagatemark(g); } else /* no more `gray' objects */ { double starttimestamp = lua_clock(); g->gcstats.currcycle.atomicstarttimestamp = starttimestamp; g->gcstats.currcycle.atomicstarttotalsizebytes = g->totalbytes; atomic(L); /* finish mark phase */ g->gcstats.currcycle.atomictime += lua_clock() - starttimestamp; } } break; } case GCSpropagateagain: { if (g->gray) { g->gcstats.currcycle.markitems++; cost = propagatemark(g); } else /* no more `gray' objects */ { double starttimestamp = lua_clock(); g->gcstats.currcycle.atomicstarttimestamp = starttimestamp; g->gcstats.currcycle.atomicstarttotalsizebytes = g->totalbytes; atomic(L); /* finish mark phase */ g->gcstats.currcycle.atomictime += lua_clock() - starttimestamp; } break; } case GCSsweepstring: { size_t traversedcount = 0; sweepwholelist(L, &g->strt.hash[g->sweepstrgc++], &traversedcount); // nothing more to sweep? if (g->sweepstrgc >= g->strt.size) { // sweep string buffer list and preserve used string count uint32_t nuse = L->global->strt.nuse; sweepwholelist(L, &g->strbufgc, &traversedcount); L->global->strt.nuse = nuse; g->gcstate = GCSsweep; // end sweep-string phase } g->gcstats.currcycle.sweepitems += traversedcount; cost = GC_SWEEPCOST; break; } case GCSsweep: { size_t traversedcount = 0; g->sweepgc = sweeplist(L, g->sweepgc, GC_SWEEPMAX, &traversedcount); g->gcstats.currcycle.sweepitems += traversedcount; if (*g->sweepgc == NULL) { /* nothing more to sweep? */ shrinkbuffers(L); g->gcstate = GCSpause; /* end collection */ } cost = GC_SWEEPMAX * GC_SWEEPCOST; break; } default: LUAU_ASSERT(0); } return cost; } static int64_t getheaptriggererroroffset(GCHeapTriggerStats* triggerstats, GCCycleStats* cyclestats) { // adjust for error using Proportional-Integral controller // https://en.wikipedia.org/wiki/PID_controller int32_t errorKb = int32_t((cyclestats->atomicstarttotalsizebytes - cyclestats->heapgoalsizebytes) / 1024); // we use sliding window for the error integral to avoid error sum 'windup' when the desired target cannot be reached int32_t* slot = &triggerstats->terms[triggerstats->termpos % triggerstats->termcount]; int32_t prev = *slot; *slot = errorKb; triggerstats->integral += errorKb - prev; triggerstats->termpos++; // controller tuning // https://en.wikipedia.org/wiki/Ziegler%E2%80%93Nichols_method const double Ku = 0.9; // ultimate gain (measured) const double Tu = 2.5; // oscillation period (measured) const double Kp = 0.45 * Ku; // proportional gain const double Ti = 0.8 * Tu; const double Ki = 0.54 * Ku / Ti; // integral gain double proportionalTerm = Kp * errorKb; double integralTerm = Ki * triggerstats->integral; double totalTerm = proportionalTerm + integralTerm; return int64_t(totalTerm * 1024); } static size_t getheaptrigger(global_State* g, size_t heapgoal) { GCCycleStats* lastcycle = &g->gcstats.lastcycle; GCCycleStats* currcycle = &g->gcstats.currcycle; // adjust threshold based on a guess of how many bytes will be allocated between the cycle start and sweep phase // our goal is to begin the sweep when used memory has reached the heap goal const double durationthreshold = 1e-3; double allocationduration = currcycle->atomicstarttimestamp - lastcycle->endtimestamp; // avoid measuring intervals smaller than 1ms if (allocationduration < durationthreshold) return heapgoal; double allocationrate = (currcycle->atomicstarttotalsizebytes - lastcycle->endtotalsizebytes) / allocationduration; double markduration = currcycle->atomicstarttimestamp - currcycle->starttimestamp; int64_t expectedgrowth = int64_t(markduration * allocationrate); int64_t offset = getheaptriggererroroffset(&g->gcstats.triggerstats, currcycle); int64_t heaptrigger = heapgoal - (expectedgrowth + offset); // clamp the trigger between memory use at the end of the cycle and the heap goal return heaptrigger < int64_t(g->totalbytes) ? g->totalbytes : (heaptrigger > int64_t(heapgoal) ? heapgoal : size_t(heaptrigger)); } void luaC_step(lua_State* L, bool assist) { global_State* g = L->global; ptrdiff_t lim = (g->gcstepsize / 100) * g->gcstepmul; /* how much to work */ LUAU_ASSERT(g->totalbytes >= g->GCthreshold); size_t debt = g->totalbytes - g->GCthreshold; GC_INTERRUPT(0); // at the start of the new cycle if (g->gcstate == GCSpause) startGcCycleStats(g); if (assist) g->gcstats.currcycle.assistwork += lim; else g->gcstats.currcycle.explicitwork += lim; int lastgcstate = g->gcstate; double lasttimestamp = lua_clock(); // always perform at least one single step do { lim -= singlestep(L); // if we have switched to a different state, capture the duration of last stage // this way we reduce the number of timer calls we make if (lastgcstate != g->gcstate) { GC_INTERRUPT(lastgcstate); double now = lua_clock(); recordGcStateTime(g, lastgcstate, now - lasttimestamp, assist); lasttimestamp = now; lastgcstate = g->gcstate; } } while (lim > 0 && g->gcstate != GCSpause); recordGcStateTime(g, lastgcstate, lua_clock() - lasttimestamp, assist); // at the end of the last cycle if (g->gcstate == GCSpause) { // at the end of a collection cycle, set goal based on gcgoal setting size_t heapgoal = (g->totalbytes / 100) * g->gcgoal; size_t heaptrigger = getheaptrigger(g, heapgoal); g->GCthreshold = heaptrigger; finishGcCycleStats(g); g->gcstats.currcycle.heapgoalsizebytes = heapgoal; g->gcstats.currcycle.heaptriggersizebytes = heaptrigger; } else { g->GCthreshold = g->totalbytes + g->gcstepsize; // compensate if GC is "behind schedule" (has some debt to pay) if (g->GCthreshold > debt) g->GCthreshold -= debt; } GC_INTERRUPT(g->gcstate); } void luaC_fullgc(lua_State* L) { global_State* g = L->global; if (g->gcstate == GCSpause) startGcCycleStats(g); if (g->gcstate <= GCSpropagateagain) { /* reset sweep marks to sweep all elements (returning them to white) */ g->sweepstrgc = 0; g->sweepgc = &g->rootgc; /* reset other collector lists */ g->gray = NULL; g->grayagain = NULL; g->weak = NULL; g->gcstate = GCSsweepstring; } LUAU_ASSERT(g->gcstate != GCSpause && g->gcstate != GCSpropagate && g->gcstate != GCSpropagateagain); /* finish any pending sweep phase */ while (g->gcstate != GCSpause) { LUAU_ASSERT(g->gcstate == GCSsweepstring || g->gcstate == GCSsweep); singlestep(L); } finishGcCycleStats(g); /* run a full collection cycle */ startGcCycleStats(g); markroot(L); while (g->gcstate != GCSpause) { singlestep(L); } /* reclaim as much buffer memory as possible (shrinkbuffers() called during sweep is incremental) */ shrinkbuffersfull(L); size_t heapgoalsizebytes = (g->totalbytes / 100) * g->gcgoal; // trigger cannot be correctly adjusted after a forced full GC. // we will try to place it so that we can reach the goal based on // the rate at which we run the GC relative to allocation rate // and on amount of bytes we need to traverse in propagation stage. // goal and stepmul are defined in percents g->GCthreshold = g->totalbytes * (g->gcgoal * g->gcstepmul / 100 - 100) / g->gcstepmul; // but it might be impossible to satisfy that directly if (g->GCthreshold < g->totalbytes) g->GCthreshold = g->totalbytes; finishGcCycleStats(g); g->gcstats.currcycle.heapgoalsizebytes = heapgoalsizebytes; g->gcstats.currcycle.heaptriggersizebytes = g->GCthreshold; } void luaC_barrierupval(lua_State* L, GCObject* v) { if (FFlag::LuauGcFullSkipInactiveThreads) { global_State* g = L->global; LUAU_ASSERT(iswhite(v) && !isdead(g, v)); if (keepinvariant(g)) reallymarkobject(g, v); } } void luaC_barrierf(lua_State* L, GCObject* o, GCObject* v) { global_State* g = L->global; LUAU_ASSERT(isblack(o) && iswhite(v) && !isdead(g, v) && !isdead(g, o)); LUAU_ASSERT(g->gcstate != GCSpause); /* must keep invariant? */ if (keepinvariant(g)) reallymarkobject(g, v); /* restore invariant */ else /* don't mind */ makewhite(g, o); /* mark as white just to avoid other barriers */ } void luaC_barriertable(lua_State* L, Table* t, GCObject* v) { global_State* g = L->global; GCObject* o = obj2gco(t); // in the second propagation stage, table assignment barrier works as a forward barrier if (FFlag::LuauRescanGrayAgainForwardBarrier && g->gcstate == GCSpropagateagain) { LUAU_ASSERT(isblack(o) && iswhite(v) && !isdead(g, v) && !isdead(g, o)); reallymarkobject(g, v); return; } LUAU_ASSERT(isblack(o) && !isdead(g, o)); LUAU_ASSERT(g->gcstate != GCSpause); black2gray(o); /* make table gray (again) */ t->gclist = g->grayagain; g->grayagain = o; } void luaC_barrierback(lua_State* L, Table* t) { global_State* g = L->global; GCObject* o = obj2gco(t); LUAU_ASSERT(isblack(o) && !isdead(g, o)); LUAU_ASSERT(g->gcstate != GCSpause); black2gray(o); /* make table gray (again) */ t->gclist = g->grayagain; g->grayagain = o; } void luaC_linkobj(lua_State* L, GCObject* o, uint8_t tt) { global_State* g = L->global; o->gch.next = g->rootgc; g->rootgc = o; o->gch.marked = luaC_white(g); o->gch.tt = tt; o->gch.memcat = L->activememcat; } void luaC_linkupval(lua_State* L, UpVal* uv) { global_State* g = L->global; GCObject* o = obj2gco(uv); o->gch.next = g->rootgc; /* link upvalue into `rootgc' list */ g->rootgc = o; if (isgray(o)) { if (keepinvariant(g)) { gray2black(o); /* closed upvalues need barrier */ luaC_barrier(L, uv, uv->v); } else { /* sweep phase: sweep it (turning it into white) */ makewhite(g, o); LUAU_ASSERT(g->gcstate != GCSpause); } } } static void validateobjref(global_State* g, GCObject* f, GCObject* t) { LUAU_ASSERT(!isdead(g, t)); if (keepinvariant(g)) { /* basic incremental invariant: black can't point to white */ LUAU_ASSERT(!(isblack(f) && iswhite(t))); } } static void validateref(global_State* g, GCObject* f, TValue* v) { if (iscollectable(v)) { LUAU_ASSERT(ttype(v) == gcvalue(v)->gch.tt); validateobjref(g, f, gcvalue(v)); } } static void validatetable(global_State* g, Table* h) { int sizenode = 1 << h->lsizenode; if (FFlag::LuauArrayBoundary) LUAU_ASSERT(h->lastfree <= sizenode); else LUAU_ASSERT(h->lastfree >= 0 && h->lastfree <= sizenode); if (h->metatable) validateobjref(g, obj2gco(h), obj2gco(h->metatable)); for (int i = 0; i < h->sizearray; ++i) validateref(g, obj2gco(h), &h->array[i]); for (int i = 0; i < sizenode; ++i) { LuaNode* n = &h->node[i]; LUAU_ASSERT(ttype(gkey(n)) != LUA_TDEADKEY || ttisnil(gval(n))); LUAU_ASSERT(i + gnext(n) >= 0 && i + gnext(n) < sizenode); if (!ttisnil(gval(n))) { TValue k = {}; k.tt = gkey(n)->tt; k.value = gkey(n)->value; validateref(g, obj2gco(h), &k); validateref(g, obj2gco(h), gval(n)); } } } static void validateclosure(global_State* g, Closure* cl) { validateobjref(g, obj2gco(cl), obj2gco(cl->env)); if (cl->isC) { for (int i = 0; i < cl->nupvalues; ++i) validateref(g, obj2gco(cl), &cl->c.upvals[i]); } else { LUAU_ASSERT(cl->nupvalues == cl->l.p->nups); validateobjref(g, obj2gco(cl), obj2gco(cl->l.p)); for (int i = 0; i < cl->nupvalues; ++i) validateref(g, obj2gco(cl), &cl->l.uprefs[i]); } } static void validatestack(global_State* g, lua_State* l) { validateref(g, obj2gco(l), gt(l)); for (CallInfo* ci = l->base_ci; ci <= l->ci; ++ci) { LUAU_ASSERT(l->stack <= ci->base); LUAU_ASSERT(ci->func <= ci->base && ci->base <= ci->top); LUAU_ASSERT(ci->top <= l->stack_last); } // note: stack refs can violate gc invariant so we only check for liveness for (StkId o = l->stack; o < l->top; ++o) checkliveness(g, o); if (l->namecall) validateobjref(g, obj2gco(l), obj2gco(l->namecall)); for (GCObject* uv = l->openupval; uv; uv = uv->gch.next) { LUAU_ASSERT(uv->gch.tt == LUA_TUPVAL); LUAU_ASSERT(gco2uv(uv)->v != &gco2uv(uv)->u.value); } } static void validateproto(global_State* g, Proto* f) { if (f->source) validateobjref(g, obj2gco(f), obj2gco(f->source)); if (f->debugname) validateobjref(g, obj2gco(f), obj2gco(f->debugname)); for (int i = 0; i < f->sizek; ++i) validateref(g, obj2gco(f), &f->k[i]); for (int i = 0; i < f->sizeupvalues; ++i) if (f->upvalues[i]) validateobjref(g, obj2gco(f), obj2gco(f->upvalues[i])); for (int i = 0; i < f->sizep; ++i) if (f->p[i]) validateobjref(g, obj2gco(f), obj2gco(f->p[i])); for (int i = 0; i < f->sizelocvars; i++) if (f->locvars[i].varname) validateobjref(g, obj2gco(f), obj2gco(f->locvars[i].varname)); } static void validateobj(global_State* g, GCObject* o) { /* dead objects can only occur during sweep */ if (isdead(g, o)) { LUAU_ASSERT(g->gcstate == GCSsweepstring || g->gcstate == GCSsweep); return; } switch (o->gch.tt) { case LUA_TSTRING: break; case LUA_TTABLE: validatetable(g, gco2h(o)); break; case LUA_TFUNCTION: validateclosure(g, gco2cl(o)); break; case LUA_TUSERDATA: if (gco2u(o)->metatable) validateobjref(g, o, obj2gco(gco2u(o)->metatable)); break; case LUA_TTHREAD: validatestack(g, gco2th(o)); break; case LUA_TPROTO: validateproto(g, gco2p(o)); break; case LUA_TUPVAL: validateref(g, o, gco2uv(o)->v); break; default: LUAU_ASSERT(!"unexpected object type"); } } static void validatelist(global_State* g, GCObject* o) { while (o) { validateobj(g, o); o = o->gch.next; } } static void validategraylist(global_State* g, GCObject* o) { if (!keepinvariant(g)) return; while (o) { LUAU_ASSERT(isgray(o)); switch (o->gch.tt) { case LUA_TTABLE: o = gco2h(o)->gclist; break; case LUA_TFUNCTION: o = gco2cl(o)->gclist; break; case LUA_TTHREAD: o = gco2th(o)->gclist; break; case LUA_TPROTO: o = gco2p(o)->gclist; break; default: LUAU_ASSERT(!"unknown object in gray list"); return; } } } void luaC_validate(lua_State* L) { global_State* g = L->global; LUAU_ASSERT(!isdead(g, obj2gco(g->mainthread))); checkliveness(g, &g->registry); for (int i = 0; i < LUA_T_COUNT; ++i) if (g->mt[i]) LUAU_ASSERT(!isdead(g, obj2gco(g->mt[i]))); validategraylist(g, g->weak); validategraylist(g, g->gray); validategraylist(g, g->grayagain); for (int i = 0; i < g->strt.size; ++i) validatelist(g, g->strt.hash[i]); validatelist(g, g->rootgc); validatelist(g, g->strbufgc); for (UpVal* uv = g->uvhead.u.l.next; uv != &g->uvhead; uv = uv->u.l.next) { LUAU_ASSERT(uv->tt == LUA_TUPVAL); LUAU_ASSERT(uv->v != &uv->u.value); LUAU_ASSERT(uv->u.l.next->u.l.prev == uv && uv->u.l.prev->u.l.next == uv); } } inline bool safejson(char ch) { return unsigned(ch) < 128 && ch >= 32 && ch != '\\' && ch != '\"'; } static void dumpref(FILE* f, GCObject* o) { fprintf(f, "\"%p\"", o); } static void dumprefs(FILE* f, TValue* data, size_t size) { bool first = true; for (size_t i = 0; i < size; ++i) { if (iscollectable(&data[i])) { if (!first) fputc(',', f); first = false; dumpref(f, gcvalue(&data[i])); } } } static void dumpstringdata(FILE* f, const char* data, size_t len) { for (size_t i = 0; i < len; ++i) fputc(safejson(data[i]) ? data[i] : '?', f); } static void dumpstring(FILE* f, TString* ts) { fprintf(f, "{\"type\":\"string\",\"cat\":%d,\"size\":%d,\"data\":\"", ts->memcat, int(sizestring(ts->len))); dumpstringdata(f, ts->data, ts->len); fprintf(f, "\"}"); } static void dumptable(FILE* f, Table* h) { size_t size = sizeof(Table) + (h->node == &luaH_dummynode ? 0 : sizenode(h) * sizeof(LuaNode)) + h->sizearray * sizeof(TValue); fprintf(f, "{\"type\":\"table\",\"cat\":%d,\"size\":%d", h->memcat, int(size)); if (h->node != &luaH_dummynode) { fprintf(f, ",\"pairs\":["); bool first = true; for (int i = 0; i < sizenode(h); ++i) { const LuaNode& n = h->node[i]; if (!ttisnil(&n.val) && (iscollectable(&n.key) || iscollectable(&n.val))) { if (!first) fputc(',', f); first = false; if (iscollectable(&n.key)) dumpref(f, gcvalue(&n.key)); else fprintf(f, "null"); fputc(',', f); if (iscollectable(&n.val)) dumpref(f, gcvalue(&n.val)); else fprintf(f, "null"); } } fprintf(f, "]"); } if (h->sizearray) { fprintf(f, ",\"array\":["); dumprefs(f, h->array, h->sizearray); fprintf(f, "]"); } if (h->metatable) { fprintf(f, ",\"metatable\":"); dumpref(f, obj2gco(h->metatable)); } fprintf(f, "}"); } static void dumpclosure(FILE* f, Closure* cl) { fprintf(f, "{\"type\":\"function\",\"cat\":%d,\"size\":%d", cl->memcat, cl->isC ? int(sizeCclosure(cl->nupvalues)) : int(sizeLclosure(cl->nupvalues))); fprintf(f, ",\"env\":"); dumpref(f, obj2gco(cl->env)); if (cl->isC) { if (cl->nupvalues) { fprintf(f, ",\"upvalues\":["); dumprefs(f, cl->c.upvals, cl->nupvalues); fprintf(f, "]"); } } else { fprintf(f, ",\"proto\":"); dumpref(f, obj2gco(cl->l.p)); if (cl->nupvalues) { fprintf(f, ",\"upvalues\":["); dumprefs(f, cl->l.uprefs, cl->nupvalues); fprintf(f, "]"); } } fprintf(f, "}"); } static void dumpudata(FILE* f, Udata* u) { fprintf(f, "{\"type\":\"userdata\",\"cat\":%d,\"size\":%d,\"tag\":%d", u->memcat, int(sizeudata(u->len)), u->tag); if (u->metatable) { fprintf(f, ",\"metatable\":"); dumpref(f, obj2gco(u->metatable)); } fprintf(f, "}"); } static void dumpthread(FILE* f, lua_State* th) { size_t size = sizeof(lua_State) + sizeof(TValue) * th->stacksize + sizeof(CallInfo) * th->size_ci; fprintf(f, "{\"type\":\"thread\",\"cat\":%d,\"size\":%d", th->memcat, int(size)); if (iscollectable(&th->l_gt)) { fprintf(f, ",\"env\":"); dumpref(f, gcvalue(&th->l_gt)); } Closure* tcl = 0; for (CallInfo* ci = th->base_ci; ci <= th->ci; ++ci) { if (ttisfunction(ci->func)) { tcl = clvalue(ci->func); break; } } if (tcl && !tcl->isC && tcl->l.p->source) { Proto* p = tcl->l.p; fprintf(f, ",\"source\":\""); dumpstringdata(f, p->source->data, p->source->len); fprintf(f, "\",\"line\":%d", p->abslineinfo ? p->abslineinfo[0] : 0); } if (th->top > th->stack) { fprintf(f, ",\"stack\":["); dumprefs(f, th->stack, th->top - th->stack); fprintf(f, "]"); } fprintf(f, "}"); } static void dumpproto(FILE* f, Proto* p) { size_t size = sizeof(Proto) + sizeof(Instruction) * p->sizecode + sizeof(Proto*) * p->sizep + sizeof(TValue) * p->sizek + p->sizelineinfo + sizeof(LocVar) * p->sizelocvars + sizeof(TString*) * p->sizeupvalues; fprintf(f, "{\"type\":\"proto\",\"cat\":%d,\"size\":%d", p->memcat, int(size)); if (p->source) { fprintf(f, ",\"source\":\""); dumpstringdata(f, p->source->data, p->source->len); fprintf(f, "\",\"line\":%d", p->abslineinfo ? p->abslineinfo[0] : 0); } if (p->sizek) { fprintf(f, ",\"constants\":["); dumprefs(f, p->k, p->sizek); fprintf(f, "]"); } if (p->sizep) { fprintf(f, ",\"protos\":["); for (int i = 0; i < p->sizep; ++i) { if (i != 0) fputc(',', f); dumpref(f, obj2gco(p->p[i])); } fprintf(f, "]"); } fprintf(f, "}"); } static void dumpupval(FILE* f, UpVal* uv) { fprintf(f, "{\"type\":\"upvalue\",\"cat\":%d,\"size\":%d", uv->memcat, int(sizeof(UpVal))); if (iscollectable(uv->v)) { fprintf(f, ",\"object\":"); dumpref(f, gcvalue(uv->v)); } fprintf(f, "}"); } static void dumpobj(FILE* f, GCObject* o) { switch (o->gch.tt) { case LUA_TSTRING: return dumpstring(f, gco2ts(o)); case LUA_TTABLE: return dumptable(f, gco2h(o)); case LUA_TFUNCTION: return dumpclosure(f, gco2cl(o)); case LUA_TUSERDATA: return dumpudata(f, gco2u(o)); case LUA_TTHREAD: return dumpthread(f, gco2th(o)); case LUA_TPROTO: return dumpproto(f, gco2p(o)); case LUA_TUPVAL: return dumpupval(f, gco2uv(o)); default: LUAU_ASSERT(0); } } static void dumplist(FILE* f, GCObject* o) { while (o) { dumpref(f, o); fputc(':', f); dumpobj(f, o); fputc(',', f); fputc('\n', f); // thread has additional list containing collectable objects that are not present in rootgc if (o->gch.tt == LUA_TTHREAD) dumplist(f, gco2th(o)->openupval); o = o->gch.next; } } void luaC_dump(lua_State* L, void* file, const char* (*categoryName)(lua_State* L, uint8_t memcat)) { global_State* g = L->global; FILE* f = static_cast(file); fprintf(f, "{\"objects\":{\n"); dumplist(f, g->rootgc); dumplist(f, g->strbufgc); for (int i = 0; i < g->strt.size; ++i) dumplist(f, g->strt.hash[i]); fprintf(f, "\"0\":{\"type\":\"userdata\",\"cat\":0,\"size\":0}\n"); // to avoid issues with trailing , fprintf(f, "},\"roots\":{\n"); fprintf(f, "\"mainthread\":"); dumpref(f, obj2gco(g->mainthread)); fprintf(f, ",\"registry\":"); dumpref(f, gcvalue(&g->registry)); fprintf(f, "},\"stats\":{\n"); fprintf(f, "\"size\":%d,\n", int(g->totalbytes)); fprintf(f, "\"categories\":{\n"); for (int i = 0; i < LUA_MEMORY_CATEGORIES; i++) { if (size_t bytes = g->memcatbytes[i]) { if (categoryName) fprintf(f, "\"%d\":{\"name\":\"%s\", \"size\":%d},\n", i, categoryName(L, i), int(bytes)); else fprintf(f, "\"%d\":{\"size\":%d},\n", i, int(bytes)); } } fprintf(f, "\"none\":{}\n"); // to avoid issues with trailing , fprintf(f, "}\n"); fprintf(f, "}}\n"); } // measure the allocation rate in bytes/sec // returns -1 if allocation rate cannot be measured int64_t luaC_allocationrate(lua_State* L) { global_State* g = L->global; const double durationthreshold = 1e-3; // avoid measuring intervals smaller than 1ms if (g->gcstate <= GCSpropagateagain) { double duration = lua_clock() - g->gcstats.lastcycle.endtimestamp; if (duration < durationthreshold) return -1; return int64_t((g->totalbytes - g->gcstats.lastcycle.endtotalsizebytes) / duration); } // totalbytes is unstable during the sweep, use the rate measured at the end of mark phase double duration = g->gcstats.currcycle.atomicstarttimestamp - g->gcstats.lastcycle.endtimestamp; if (duration < durationthreshold) return -1; return int64_t((g->gcstats.currcycle.atomicstarttotalsizebytes - g->gcstats.lastcycle.endtotalsizebytes) / duration); } void luaC_wakethread(lua_State* L) { if (!luaC_threadsleeping(L)) return; global_State* g = L->global; resetbit(L->stackstate, THREAD_SLEEPINGBIT); if (keepinvariant(g)) { GCObject* o = obj2gco(L); L->gclist = g->grayagain; g->grayagain = o; black2gray(o); } } const char* luaC_statename(int state) { switch (state) { case GCSpause: return "pause"; case GCSpropagate: return "mark"; case GCSpropagateagain: return "remark"; case GCSatomic: return "atomic"; case GCSsweepstring: return "sweepstring"; case GCSsweep: return "sweep"; default: return NULL; } }