Github User Fetcher 1.0.0
C Application with Server and GUI
Loading...
Searching...
No Matches
lgc.c File Reference
#include <string.h>
#include "lua.h"
#include "ldebug.h"
#include "ldo.h"
#include "lfunc.h"
#include "lgc.h"
#include "lmem.h"
#include "lobject.h"
#include "lstate.h"
#include "lstring.h"
#include "ltable.h"
#include "ltm.h"

Go to the source code of this file.

Macros

#define lgc_c
 
#define LUA_CORE
 
#define GCSTEPSIZE   1024u
 
#define GCSWEEPMAX   40
 
#define GCSWEEPCOST   10
 
#define GCFINALIZECOST   100
 
#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)->tsv.marked, WHITE0BIT, WHITE1BIT)
 
#define isfinalized(u)   testbit((u)->marked, FINALIZEDBIT)
 
#define markfinalized(u)   l_setbit((u)->marked, FINALIZEDBIT)
 
#define KEYWEAK   bitmask(KEYWEAKBIT)
 
#define VALUEWEAK   bitmask(VALUEWEAKBIT)
 
#define markvalue(g, o)
 
#define markobject(g, t)
 
#define setthreshold(g)   (g->GCthreshold = (g->estimate/100) * g->gcpause)
 
#define sweepwholelist(L, p)   sweeplist(L,p,MAX_LUMEM)
 

Functions

static void removeentry (Node *n)
 
static void reallymarkobject (global_State *g, GCObject *o)
 
static void marktmu (global_State *g)
 
size_t luaC_separateudata (lua_State *L, int all)
 
static int traversetable (global_State *g, Table *h)
 
static void traverseproto (global_State *g, Proto *f)
 
static void traverseclosure (global_State *g, Closure *cl)
 
static void checkstacksizes (lua_State *L, StkId max)
 
static void traversestack (global_State *g, lua_State *l)
 
static l_mem propagatemark (global_State *g)
 
static size_t propagateall (global_State *g)
 
static int iscleared (const TValue *o, int iskey)
 
static void cleartable (GCObject *l)
 
static void freeobj (lua_State *L, GCObject *o)
 
static GCObject ** sweeplist (lua_State *L, GCObject **p, lu_mem count)
 
static void checkSizes (lua_State *L)
 
static void GCTM (lua_State *L)
 
void luaC_callGCTM (lua_State *L)
 
void luaC_freeall (lua_State *L)
 
static void markmt (global_State *g)
 
static void markroot (lua_State *L)
 
static void remarkupvals (global_State *g)
 
static void atomic (lua_State *L)
 
static l_mem singlestep (lua_State *L)
 
void luaC_step (lua_State *L)
 
void luaC_fullgc (lua_State *L)
 
void luaC_barrierf (lua_State *L, GCObject *o, GCObject *v)
 
void luaC_barrierback (lua_State *L, Table *t)
 
void luaC_link (lua_State *L, GCObject *o, lu_byte tt)
 
void luaC_linkupval (lua_State *L, UpVal *uv)
 

Macro Definition Documentation

◆ black2gray

#define black2gray ( x)    resetbit((x)->gch.marked, BLACKBIT)

Definition at line 38 of file lua-5.1.5/src/lgc.c.

Referenced by luaC_barrierback(), and propagatemark().

◆ GCFINALIZECOST

#define GCFINALIZECOST   100

Definition at line 29 of file lua-5.1.5/src/lgc.c.

Referenced by singlestep().

◆ GCSTEPSIZE

#define GCSTEPSIZE   1024u

Definition at line 26 of file lua-5.1.5/src/lgc.c.

Referenced by incstep(), lua_gc(), and luaC_step().

◆ GCSWEEPCOST

#define GCSWEEPCOST   10

Definition at line 28 of file lua-5.1.5/src/lgc.c.

Referenced by singlestep().

◆ GCSWEEPMAX

#define GCSWEEPMAX   40

Definition at line 27 of file lua-5.1.5/src/lgc.c.

Referenced by singlestep().

◆ isfinalized

#define isfinalized ( u)    testbit((u)->marked, FINALIZEDBIT)

Definition at line 43 of file lua-5.1.5/src/lgc.c.

Referenced by iscleared(), and luaC_separateudata().

◆ KEYWEAK

#define KEYWEAK   bitmask(KEYWEAKBIT)

Definition at line 47 of file lua-5.1.5/src/lgc.c.

Referenced by traversetable().

◆ lgc_c

#define lgc_c

Definition at line 9 of file lua-5.1.5/src/lgc.c.

◆ LUA_CORE

#define LUA_CORE

Definition at line 10 of file lua-5.1.5/src/lgc.c.

◆ makewhite

#define makewhite ( g,
x )    ((x)->gch.marked = cast_byte(((x)->gch.marked & maskmarks) | luaC_white(g)))

Definition at line 34 of file lua-5.1.5/src/lgc.c.

34#define makewhite(g,x) \
35 ((x)->gch.marked = cast_byte(((x)->gch.marked & maskmarks) | luaC_white(g)))

Referenced by GCTM(), luaC_barrierf(), luaC_linkupval(), marktmu(), and sweeplist().

◆ markfinalized

#define markfinalized ( u)    l_setbit((u)->marked, FINALIZEDBIT)

Definition at line 44 of file lua-5.1.5/src/lgc.c.

Referenced by luaC_separateudata().

◆ markobject

#define markobject ( g,
t )
Value:
{ if (iswhite(obj2gco(t))) \
reallymarkobject(g, obj2gco(t)); }
#define iswhite(x)
#define obj2gco(v)

Definition at line 55 of file lua-5.1.5/src/lgc.c.

55#define markobject(g,t) { if (iswhite(obj2gco(t))) \
56 reallymarkobject(g, obj2gco(t)); }

Referenced by atomic(), markmt(), markroot(), reallymarkobject(), traverseclosure(), traverseproto(), and traversetable().

◆ markvalue

#define markvalue ( g,
o )
Value:
static void reallymarkobject(global_State *g, GCObject *o)
#define gcvalue(o)
#define iscollectable(o)
#define checkconsistency(obj)

Definition at line 52 of file lua-5.1.5/src/lgc.c.

52#define markvalue(g,o) { checkconsistency(o); \
53 if (iscollectable(o) && iswhite(gcvalue(o))) reallymarkobject(g,gcvalue(o)); }

Referenced by markroot(), reallymarkobject(), remarkupvals(), traverseclosure(), traverseproto(), traversestack(), and traversetable().

◆ maskmarks

#define maskmarks   cast_byte(~(bitmask(BLACKBIT)|WHITEBITS))

Definition at line 32 of file lua-5.1.5/src/lgc.c.

◆ setthreshold

#define setthreshold ( g)    (g->GCthreshold = (g->estimate/100) * g->gcpause)

Definition at line 59 of file lua-5.1.5/src/lgc.c.

Referenced by luaC_fullgc(), and luaC_step().

◆ stringmark

#define stringmark ( s)    reset2bits((s)->tsv.marked, WHITE0BIT, WHITE1BIT)

Definition at line 40 of file lua-5.1.5/src/lgc.c.

Referenced by iscleared(), and traverseproto().

◆ sweepwholelist

#define sweepwholelist ( L,
p )   sweeplist(L,p,MAX_LUMEM)

Definition at line 404 of file lua-5.1.5/src/lgc.c.

Referenced by luaC_freeall(), singlestep(), and sweeplist().

◆ VALUEWEAK

#define VALUEWEAK   bitmask(VALUEWEAKBIT)

Definition at line 48 of file lua-5.1.5/src/lgc.c.

Referenced by traversetable().

◆ white2gray

#define white2gray ( x)    reset2bits((x)->gch.marked, WHITE0BIT, WHITE1BIT)

Definition at line 37 of file lua-5.1.5/src/lgc.c.

Referenced by reallymarkobject().

Function Documentation

◆ atomic()

static void atomic ( lua_State * L)
static

Definition at line 525 of file lua-5.1.5/src/lgc.c.

525 {
526 global_State *g = G(L);
527 size_t udsize; /* total size of userdata to be finalized */
528 /* remark occasional upvalues of (maybe) dead threads */
529 remarkupvals(g);
530 /* traverse objects cautch by write barrier and by 'remarkupvals' */
531 propagateall(g);
532 /* remark weak tables */
533 g->gray = g->weak;
534 g->weak = NULL;
536 markobject(g, L); /* mark running thread */
537 markmt(g); /* mark basic metatables (again) */
538 propagateall(g);
539 /* remark gray again */
540 g->gray = g->grayagain;
541 g->grayagain = NULL;
542 propagateall(g);
543 udsize = luaC_separateudata(L, 0); /* separate userdata to be finalized */
544 marktmu(g); /* mark `preserved' userdata */
545 udsize += propagateall(g); /* remark, to propagate `preserveness' */
546 cleartable(g->weak); /* remove collected objects from weak tables */
547 /* flip current white */
549 g->sweepstrgc = 0;
550 g->sweepgc = &g->rootgc;
552 g->estimate = g->totalbytes - udsize; /* first estimate */
553}
#define NULL
Definition gmacros.h:924
static void marktmu(global_State *g)
size_t luaC_separateudata(lua_State *L, int all)
static size_t propagateall(global_State *g)
#define markobject(g, t)
static void remarkupvals(global_State *g)
static void markmt(global_State *g)
static void cleartable(GCObject *l)
#define otherwhite(g)
#define GCSsweepstring
#define cast_byte(i)
#define lua_assert(c)
#define G(L)
struct lua_State * mainthread

References cast_byte, cleartable(), global_State::currentwhite, global_State::estimate, G, GCSsweepstring, global_State::gcstate, global_State::gray, global_State::grayagain, iswhite, lua_assert, luaC_separateudata(), global_State::mainthread, markmt(), markobject, marktmu(), NULL, obj2gco, otherwhite, propagateall(), remarkupvals(), global_State::rootgc, global_State::sweepgc, global_State::sweepstrgc, global_State::totalbytes, and global_State::weak.

Referenced by singlestep().

◆ checkSizes()

static void checkSizes ( lua_State * L)
static

Definition at line 431 of file lua-5.1.5/src/lgc.c.

431 {
432 global_State *g = G(L);
433 /* check size of string hash */
434 if (g->strt.nuse < cast(lu_int32, g->strt.size/4) &&
435 g->strt.size > MINSTRTABSIZE*2)
436 luaS_resize(L, g->strt.size/2); /* table is too big */
437 /* check size of buffer */
438 if (luaZ_sizebuffer(&g->buff) > LUA_MINBUFFER*2) { /* buffer too big? */
439 size_t newsize = luaZ_sizebuffer(&g->buff) / 2;
440 luaZ_resizebuffer(L, &g->buff, newsize);
441 }
442}
LUAI_UINT32 lu_int32
#define MINSTRTABSIZE
#define LUA_MINBUFFER
#define cast(t, exp)
void luaS_resize(lua_State *L, int newsize)
#define luaZ_sizebuffer(buff)
#define luaZ_resizebuffer(L, buff, size)

References global_State::buff, cast, G, LUA_MINBUFFER, luaS_resize(), luaZ_resizebuffer, luaZ_sizebuffer, MINSTRTABSIZE, stringtable::nuse, stringtable::size, and global_State::strt.

Referenced by singlestep().

◆ checkstacksizes()

static void checkstacksizes ( lua_State * L,
StkId max )
static

Definition at line 241 of file lua-5.1.5/src/lgc.c.

241 {
242 int ci_used = cast_int(L->ci - L->base_ci); /* number of `ci' in use */
243 int s_used = cast_int(max - L->stack); /* part of stack in use */
244 if (L->size_ci > LUAI_MAXCALLS) /* handling overflow? */
245 return; /* do not touch the stacks */
246 if (4*ci_used < L->size_ci && 2*BASIC_CI_SIZE < L->size_ci)
247 luaD_reallocCI(L, L->size_ci/2); /* still big enough... */
248 condhardstacktests(luaD_reallocCI(L, ci_used + 1));
249 if (4*s_used < L->stacksize &&
251 luaD_reallocstack(L, L->stacksize/2); /* still big enough... */
253}
void luaD_reallocCI(lua_State *L, int newsize)
void luaD_reallocstack(lua_State *L, int newsize)
#define cast_int(i)
#define condhardstacktests(x)
#define BASIC_STACK_SIZE
#define EXTRA_STACK
#define LUAI_MAXCALLS
#define stacksize(th)

References lua_State::base_ci, BASIC_STACK_SIZE, cast_int, lua_State::ci, condhardstacktests, EXTRA_STACK, luaD_reallocCI(), luaD_reallocstack(), LUAI_MAXCALLS, lua_State::size_ci, lua_State::stack, lua_State::stacksize, and stacksize.

Referenced by traversestack().

◆ cleartable()

static void cleartable ( GCObject * l)
static

Definition at line 351 of file lua-5.1.5/src/lgc.c.

351 {
352 while (l) {
353 Table *h = gco2h(l);
354 int i = h->sizearray;
355 lua_assert(testbit(h->marked, VALUEWEAKBIT) ||
356 testbit(h->marked, KEYWEAKBIT));
357 if (testbit(h->marked, VALUEWEAKBIT)) {
358 while (i--) {
359 TValue *o = &h->array[i];
360 if (iscleared(o, 0)) /* value was collected? */
361 setnilvalue(o); /* remove value */
362 }
363 }
364 i = sizenode(h);
365 while (i--) {
366 Node *n = gnode(h, i);
367 if (!ttisnil(gval(n)) && /* non-empty entry? */
368 (iscleared(key2tval(n), 1) || iscleared(gval(n), 0))) {
369 setnilvalue(gval(n)); /* remove value ... */
370 removeentry(n); /* remove entry from table */
371 }
372 }
373 l = h->gclist;
374 }
375}
static void removeentry(Node *n)
static int iscleared(const TValue *o, int iskey)
#define VALUEWEAKBIT
#define testbit(x, b)
#define KEYWEAKBIT
#define sizenode(t)
#define ttisnil(o)
#define setnilvalue(obj)
#define gco2h(o)
#define gnode(t, i)
#define gval(n)
#define key2tval(n)
GCObject * gclist

References Table::array, Table::gclist, gco2h, gnode, gval, iscleared(), key2tval, KEYWEAKBIT, lua_assert, removeentry(), setnilvalue, Table::sizearray, sizenode, testbit, ttisnil, and VALUEWEAKBIT.

Referenced by atomic().

◆ freeobj()

static void freeobj ( lua_State * L,
GCObject * o )
static

Definition at line 378 of file lua-5.1.5/src/lgc.c.

378 {
379 switch (o->gch.tt) {
380 case LUA_TPROTO: luaF_freeproto(L, gco2p(o)); break;
381 case LUA_TFUNCTION: luaF_freeclosure(L, gco2cl(o)); break;
382 case LUA_TUPVAL: luaF_freeupval(L, gco2uv(o)); break;
383 case LUA_TTABLE: luaH_free(L, gco2h(o)); break;
384 case LUA_TTHREAD: {
385 lua_assert(gco2th(o) != L && gco2th(o) != G(L)->mainthread);
386 luaE_freethread(L, gco2th(o));
387 break;
388 }
389 case LUA_TSTRING: {
390 G(L)->strt.nuse--;
391 luaM_freemem(L, o, sizestring(gco2ts(o)));
392 break;
393 }
394 case LUA_TUSERDATA: {
395 luaM_freemem(L, o, sizeudata(gco2u(o)));
396 break;
397 }
398 default: lua_assert(0);
399 }
400}
void luaF_freeproto(lua_State *L, Proto *f)
void luaF_freeclosure(lua_State *L, Closure *c)
void luaF_freeupval(lua_State *L, UpVal *uv)
#define luaM_freemem(L, b, s)
#define LUA_TUPVAL
#define LUA_TPROTO
void luaE_freethread(lua_State *L, lua_State *L1)
#define gco2cl(o)
#define gco2th(o)
#define gco2u(o)
#define gco2p(o)
#define gco2uv(o)
#define gco2ts(o)
#define sizeudata(u)
#define sizestring(s)
void luaH_free(lua_State *L, Table *t)
#define LUA_TTHREAD
#define LUA_TTABLE
#define LUA_TSTRING
#define LUA_TUSERDATA
#define LUA_TFUNCTION

References G, GCObject::gch, gco2cl, gco2h, gco2p, gco2th, gco2ts, gco2u, gco2uv, lua_assert, LUA_TFUNCTION, LUA_TPROTO, LUA_TSTRING, LUA_TTABLE, LUA_TTHREAD, LUA_TUPVAL, LUA_TUSERDATA, luaE_freethread(), luaF_freeclosure(), luaF_freeproto(), luaF_freeupval(), luaH_free(), luaM_freemem, sizestring, and sizeudata.

Referenced by sweeplist().

◆ GCTM()

static void GCTM ( lua_State * L)
static

Definition at line 445 of file lua-5.1.5/src/lgc.c.

445 {
446 global_State *g = G(L);
447 GCObject *o = g->tmudata->gch.next; /* get first element */
448 Udata *udata = rawgco2u(o);
449 const TValue *tm;
450 /* remove udata from `tmudata' */
451 if (o == g->tmudata) /* last element? */
452 g->tmudata = NULL;
453 else
454 g->tmudata->gch.next = udata->uv.next;
455 udata->uv.next = g->mainthread->next; /* return it to `root' list */
456 g->mainthread->next = o;
457 makewhite(g, o);
458 tm = fasttm(L, udata->uv.metatable, TM_GC);
459 if (tm != NULL) {
460 lu_byte oldah = L->allowhook;
461 lu_mem oldt = g->GCthreshold;
462 L->allowhook = 0; /* stop debug hooks during GC tag method */
463 g->GCthreshold = 2*g->totalbytes; /* avoid GC steps */
464 setobj2s(L, L->top, tm);
465 setuvalue(L, L->top+1, udata);
466 L->top += 2;
467 luaD_call(L, L->top - 2, 0);
468 L->allowhook = oldah; /* restore hooks */
469 g->GCthreshold = oldt; /* restore threshold */
470 }
471}
void luaD_call(lua_State *L, StkId func, int nResults)
#define makewhite(g, x)
LUAI_UMEM lu_mem
unsigned char lu_byte
#define setobj2s
#define setuvalue(L, obj, x)
#define rawgco2u(o)
#define fasttm(l, et, e)
@ TM_GC
struct Udata::@47 uv
struct Table * metatable

References lua_State::allowhook, fasttm, G, GCObject::gch, global_State::GCthreshold, luaD_call(), global_State::mainthread, makewhite, Udata::metatable, NULL, rawgco2u, setobj2s, setuvalue, TM_GC, global_State::tmudata, lua_State::top, global_State::totalbytes, and Udata::uv.

Referenced by luaC_callGCTM(), and singlestep().

◆ iscleared()

static int iscleared ( const TValue * o,
int iskey )
static

Definition at line 337 of file lua-5.1.5/src/lgc.c.

337 {
338 if (!iscollectable(o)) return 0;
339 if (ttisstring(o)) {
340 stringmark(rawtsvalue(o)); /* strings are `values', so are never weak */
341 return 0;
342 }
343 return iswhite(gcvalue(o)) ||
344 (ttisuserdata(o) && (!iskey && isfinalized(uvalue(o))));
345}
#define isfinalized(u)
#define stringmark(s)
#define uvalue(o)
#define ttisstring(o)
#define rawtsvalue(o)
#define ttisuserdata(o)

References gcvalue, iscollectable, isfinalized, iswhite, rawtsvalue, stringmark, ttisstring, ttisuserdata, and uvalue.

Referenced by cleartable().

◆ luaC_barrierback()

void luaC_barrierback ( lua_State * L,
Table * t )

Definition at line 674 of file lua-5.1.5/src/lgc.c.

674 {
675 global_State *g = G(L);
676 GCObject *o = obj2gco(t);
677 lua_assert(isblack(o) && !isdead(g, o));
679 black2gray(o); /* make table gray (again) */
680 t->gclist = g->grayagain;
681 g->grayagain = o;
682}
#define black2gray(x)
#define GCSpause
#define GCSfinalize
#define isblack(x)
#define isdead(g, v)

References black2gray, G, Table::gclist, GCSfinalize, GCSpause, global_State::gcstate, global_State::grayagain, isblack, isdead, lua_assert, and obj2gco.

◆ luaC_barrierf()

void luaC_barrierf ( lua_State * L,
GCObject * o,
GCObject * v )

Definition at line 661 of file lua-5.1.5/src/lgc.c.

661 {
662 global_State *g = G(L);
663 lua_assert(isblack(o) && iswhite(v) && !isdead(g, v) && !isdead(g, o));
666 /* must keep invariant? */
667 if (g->gcstate == GCSpropagate)
668 reallymarkobject(g, v); /* restore invariant */
669 else /* don't mind */
670 makewhite(g, o); /* mark as white just to avoid other barriers */
671}
#define GCSpropagate
#define ttype(o)

References G, GCObject::gch, GCSfinalize, GCSpause, GCSpropagate, global_State::gcstate, isblack, isdead, iswhite, lua_assert, LUA_TTABLE, makewhite, reallymarkobject(), and ttype.

◆ luaC_callGCTM()

void luaC_callGCTM ( lua_State * L)

Definition at line 477 of file lua-5.1.5/src/lgc.c.

477 {
478 while (G(L)->tmudata)
479 GCTM(L);
480}
static void GCTM(lua_State *L)

References G, and GCTM().

Referenced by callallgcTM().

◆ luaC_freeall()

void luaC_freeall ( lua_State * L)

Definition at line 483 of file lua-5.1.5/src/lgc.c.

483 {
484 global_State *g = G(L);
485 int i;
486 g->currentwhite = WHITEBITS | bitmask(SFIXEDBIT); /* mask to collect all elements */
487 sweepwholelist(L, &g->rootgc);
488 for (i = 0; i < g->strt.size; i++) /* free all string lists */
489 sweepwholelist(L, &g->strt.hash[i]);
490}
#define sweepwholelist(L, p)
#define WHITEBITS
#define SFIXEDBIT
#define bitmask(b)

References bitmask, global_State::currentwhite, G, stringtable::hash, global_State::rootgc, SFIXEDBIT, stringtable::size, global_State::strt, sweepwholelist, and WHITEBITS.

Referenced by close_state().

◆ luaC_fullgc()

void luaC_fullgc ( lua_State * L)

Definition at line 635 of file lua-5.1.5/src/lgc.c.

635 {
636 global_State *g = G(L);
637 if (g->gcstate <= GCSpropagate) {
638 /* reset sweep marks to sweep all elements (returning them to white) */
639 g->sweepstrgc = 0;
640 g->sweepgc = &g->rootgc;
641 /* reset other collector lists */
642 g->gray = NULL;
643 g->grayagain = NULL;
644 g->weak = NULL;
646 }
648 /* finish any pending sweep phase */
649 while (g->gcstate != GCSfinalize) {
651 singlestep(L);
652 }
653 markroot(L);
654 while (g->gcstate != GCSpause) {
655 singlestep(L);
656 }
657 setthreshold(g);
658}
static void markroot(lua_State *L)
static l_mem singlestep(lua_State *L)
#define setthreshold(g)
#define GCSsweep

References G, GCSfinalize, GCSpause, GCSpropagate, GCSsweep, GCSsweepstring, global_State::gcstate, global_State::gray, global_State::grayagain, lua_assert, markroot(), NULL, global_State::rootgc, setthreshold, singlestep(), global_State::sweepgc, global_State::sweepstrgc, and global_State::weak.

Referenced by growstrtab(), lua_gc(), lua_gc(), luaM_realloc_(), and tryagain().

◆ luaC_link()

void luaC_link ( lua_State * L,
GCObject * o,
lu_byte tt )

Definition at line 685 of file lua-5.1.5/src/lgc.c.

685 {
686 global_State *g = G(L);
687 o->gch.next = g->rootgc;
688 g->rootgc = o;
689 o->gch.marked = luaC_white(g);
690 o->gch.tt = tt;
691}
#define luaC_white(g)

References G, GCObject::gch, luaC_white, and global_State::rootgc.

Referenced by luaE_newthread(), luaF_newCclosure(), luaF_newLclosure(), luaF_newproto(), luaF_newupval(), and luaH_new().

◆ luaC_linkupval()

void luaC_linkupval ( lua_State * L,
UpVal * uv )

Definition at line 694 of file lua-5.1.5/src/lgc.c.

694 {
695 global_State *g = G(L);
696 GCObject *o = obj2gco(uv);
697 o->gch.next = g->rootgc; /* link upvalue into `rootgc' list */
698 g->rootgc = o;
699 if (isgray(o)) {
700 if (g->gcstate == GCSpropagate) {
701 gray2black(o); /* closed upvalues need barrier */
702 luaC_barrier(L, uv, uv->v);
703 }
704 else { /* sweep phase: sweep it (turning it into white) */
705 makewhite(g, o);
707 }
708 }
709}
#define luaC_barrier(L, p, v)
#define isgray(x)
#define gray2black(x)

References G, GCObject::gch, GCSfinalize, GCSpause, GCSpropagate, global_State::gcstate, gray2black, isgray, lua_assert, luaC_barrier, makewhite, obj2gco, global_State::rootgc, and UpVal::v.

Referenced by luaF_close().

◆ luaC_separateudata()

size_t luaC_separateudata ( lua_State * L,
int all )

Definition at line 128 of file lua-5.1.5/src/lgc.c.

128 {
129 global_State *g = G(L);
130 size_t deadmem = 0;
131 GCObject **p = &g->mainthread->next;
132 GCObject *curr;
133 while ((curr = *p) != NULL) {
134 if (!(iswhite(curr) || all) || isfinalized(gco2u(curr)))
135 p = &curr->gch.next; /* don't bother with them */
136 else if (fasttm(L, gco2u(curr)->metatable, TM_GC) == NULL) {
137 markfinalized(gco2u(curr)); /* don't need finalization */
138 p = &curr->gch.next;
139 }
140 else { /* must call its gc method */
141 deadmem += sizeudata(gco2u(curr));
142 markfinalized(gco2u(curr));
143 *p = curr->gch.next;
144 /* link `curr' at the end of `tmudata' list */
145 if (g->tmudata == NULL) /* list is empty? */
146 g->tmudata = curr->gch.next = curr; /* creates a circular list */
147 else {
148 curr->gch.next = g->tmudata->gch.next;
149 g->tmudata->gch.next = curr;
150 g->tmudata = curr;
151 }
152 }
153 }
154 return deadmem;
155}
#define markfinalized(u)

References fasttm, G, GCObject::gch, gco2u, isfinalized, iswhite, global_State::mainthread, markfinalized, NULL, sizeudata, TM_GC, and global_State::tmudata.

Referenced by atomic(), and lua_close().

◆ luaC_step()

void luaC_step ( lua_State * L)

Definition at line 610 of file lua-5.1.5/src/lgc.c.

610 {
611 global_State *g = G(L);
612 l_mem lim = (GCSTEPSIZE/100) * g->gcstepmul;
613 if (lim == 0)
614 lim = (MAX_LUMEM-1)/2; /* no limit */
615 g->gcdept += g->totalbytes - g->GCthreshold;
616 do {
617 lim -= singlestep(L);
618 if (g->gcstate == GCSpause)
619 break;
620 } while (lim > 0);
621 if (g->gcstate != GCSpause) {
622 if (g->gcdept < GCSTEPSIZE)
623 g->GCthreshold = g->totalbytes + GCSTEPSIZE; /* - lim/g->gcstepmul;*/
624 else {
625 g->gcdept -= GCSTEPSIZE;
626 g->GCthreshold = g->totalbytes;
627 }
628 }
629 else {
630 setthreshold(g);
631 }
632}
#define GCSTEPSIZE
#define MAX_LUMEM
LUAI_MEM l_mem

References G, global_State::gcdept, GCSpause, global_State::gcstate, global_State::gcstepmul, GCSTEPSIZE, global_State::GCthreshold, MAX_LUMEM, setthreshold, singlestep(), and global_State::totalbytes.

Referenced by lua_gc(), and lua_gc().

◆ markmt()

static void markmt ( global_State * g)
static

Definition at line 493 of file lua-5.1.5/src/lgc.c.

493 {
494 int i;
495 for (i=0; i<NUM_TAGS; i++)
496 if (g->mt[i]) markobject(g, g->mt[i]);
497}
#define NUM_TAGS
struct Table * mt[NUM_TAGS]

References markobject, global_State::mt, and NUM_TAGS.

Referenced by atomic(), and markroot().

◆ markroot()

static void markroot ( lua_State * L)
static

Definition at line 501 of file lua-5.1.5/src/lgc.c.

501 {
502 global_State *g = G(L);
503 g->gray = NULL;
504 g->grayagain = NULL;
505 g->weak = NULL;
506 markobject(g, g->mainthread);
507 /* make global table be traversed before main stack */
508 markvalue(g, gt(g->mainthread));
509 markvalue(g, registry(L));
510 markmt(g);
512}
#define markvalue(g, o)
#define gt(L)
#define registry(L)

References G, GCSpropagate, global_State::gcstate, global_State::gray, global_State::grayagain, gt, global_State::mainthread, markmt(), markobject, markvalue, NULL, registry, and global_State::weak.

Referenced by luaC_fullgc(), and singlestep().

◆ marktmu()

static void marktmu ( global_State * g)
static

Definition at line 115 of file lua-5.1.5/src/lgc.c.

115 {
116 GCObject *u = g->tmudata;
117 if (u) {
118 do {
119 u = u->gch.next;
120 makewhite(g, u); /* may be marked, if left from previous GC */
121 reallymarkobject(g, u);
122 } while (u != g->tmudata);
123 }
124}

References GCObject::gch, makewhite, reallymarkobject(), and global_State::tmudata.

Referenced by atomic().

◆ propagateall()

static size_t propagateall ( global_State * g)
static

Definition at line 323 of file lua-5.1.5/src/lgc.c.

323 {
324 size_t m = 0;
325 while (g->gray) m += propagatemark(g);
326 return m;
327}
static l_mem propagatemark(global_State *g)

References global_State::gray, and propagatemark().

Referenced by atomic().

◆ propagatemark()

static l_mem propagatemark ( global_State * g)
static

Definition at line 277 of file lua-5.1.5/src/lgc.c.

277 {
278 GCObject *o = g->gray;
279 lua_assert(isgray(o));
280 gray2black(o);
281 switch (o->gch.tt) {
282 case LUA_TTABLE: {
283 Table *h = gco2h(o);
284 g->gray = h->gclist;
285 if (traversetable(g, h)) /* table is weak? */
286 black2gray(o); /* keep it gray */
287 return sizeof(Table) + sizeof(TValue) * h->sizearray +
288 sizeof(Node) * sizenode(h);
289 }
290 case LUA_TFUNCTION: {
291 Closure *cl = gco2cl(o);
292 g->gray = cl->c.gclist;
293 traverseclosure(g, cl);
294 return (cl->c.isC) ? sizeCclosure(cl->c.nupvalues) :
295 sizeLclosure(cl->l.nupvalues);
296 }
297 case LUA_TTHREAD: {
298 lua_State *th = gco2th(o);
299 g->gray = th->gclist;
300 th->gclist = g->grayagain;
301 g->grayagain = o;
302 black2gray(o);
303 traversestack(g, th);
304 return sizeof(lua_State) + sizeof(TValue) * th->stacksize +
305 sizeof(CallInfo) * th->size_ci;
306 }
307 case LUA_TPROTO: {
308 Proto *p = gco2p(o);
309 g->gray = p->gclist;
310 traverseproto(g, p);
311 return sizeof(Proto) + sizeof(Instruction) * p->sizecode +
312 sizeof(Proto *) * p->sizep +
313 sizeof(TValue) * p->sizek +
314 sizeof(int) * p->sizelineinfo +
315 sizeof(LocVar) * p->sizelocvars +
316 sizeof(TString *) * p->sizeupvalues;
317 }
318 default: lua_assert(0); return 0;
319 }
320}
#define sizeLclosure(n)
#define sizeCclosure(n)
static void traverseproto(global_State *g, Proto *f)
static void traversestack(global_State *g, lua_State *l)
static int traversetable(global_State *g, Table *h)
static void traverseclosure(global_State *g, Closure *cl)
lu_int32 Instruction
struct Node Node
struct Table Table
struct Proto Proto
struct CallInfo CallInfo
struct lua_State lua_State
GCObject * gclist

References black2gray, Closure::c, GCObject::gch, Proto::gclist, Table::gclist, lua_State::gclist, gco2cl, gco2h, gco2p, gco2th, global_State::gray, gray2black, global_State::grayagain, isgray, Closure::l, lua_assert, LUA_TFUNCTION, LUA_TPROTO, LUA_TTABLE, LUA_TTHREAD, lua_State::size_ci, Table::sizearray, sizeCclosure, Proto::sizecode, Proto::sizek, sizeLclosure, Proto::sizelineinfo, Proto::sizelocvars, sizenode, Proto::sizep, Proto::sizeupvalues, lua_State::stacksize, traverseclosure(), traverseproto(), traversestack(), and traversetable().

Referenced by propagateall(), and singlestep().

◆ reallymarkobject()

static void reallymarkobject ( global_State * g,
GCObject * o )
static

Definition at line 69 of file lua-5.1.5/src/lgc.c.

69 {
70 lua_assert(iswhite(o) && !isdead(g, o));
71 white2gray(o);
72 switch (o->gch.tt) {
73 case LUA_TSTRING: {
74 return;
75 }
76 case LUA_TUSERDATA: {
77 Table *mt = gco2u(o)->metatable;
78 gray2black(o); /* udata are never gray */
79 if (mt) markobject(g, mt);
80 markobject(g, gco2u(o)->env);
81 return;
82 }
83 case LUA_TUPVAL: {
84 UpVal *uv = gco2uv(o);
85 markvalue(g, uv->v);
86 if (uv->v == &uv->u.value) /* closed? */
87 gray2black(o); /* open upvalues are never black */
88 return;
89 }
90 case LUA_TFUNCTION: {
91 gco2cl(o)->c.gclist = g->gray;
92 g->gray = o;
93 break;
94 }
95 case LUA_TTABLE: {
96 gco2h(o)->gclist = g->gray;
97 g->gray = o;
98 break;
99 }
100 case LUA_TTHREAD: {
101 gco2th(o)->gclist = g->gray;
102 g->gray = o;
103 break;
104 }
105 case LUA_TPROTO: {
106 gco2p(o)->gclist = g->gray;
107 g->gray = o;
108 break;
109 }
110 default: lua_assert(0);
111 }
112}
#define white2gray(x)
struct Table * metatable
union UpVal::@48 u

References GCObject::gch, gco2cl, gco2h, gco2p, gco2th, gco2u, gco2uv, global_State::gray, gray2black, isdead, iswhite, lua_assert, LUA_TFUNCTION, LUA_TPROTO, LUA_TSTRING, LUA_TTABLE, LUA_TTHREAD, LUA_TUPVAL, LUA_TUSERDATA, markobject, markvalue, Table::metatable, UpVal::u, UpVal::v, UpVal::value, and white2gray.

Referenced by luaC_barrierf(), and marktmu().

◆ remarkupvals()

static void remarkupvals ( global_State * g)
static

Definition at line 515 of file lua-5.1.5/src/lgc.c.

515 {
516 UpVal *uv;
517 for (uv = g->uvhead.u.l.next; uv != &g->uvhead; uv = uv->u.l.next) {
518 lua_assert(uv->u.l.next->u.l.prev == uv && uv->u.l.prev->u.l.next == uv);
519 if (isgray(obj2gco(uv)))
520 markvalue(g, uv->v);
521 }
522}
struct UpVal * prev
struct UpVal * next
struct UpVal::@48::@49 l

References isgray, UpVal::l, lua_assert, markvalue, UpVal::next, obj2gco, UpVal::prev, UpVal::u, global_State::uvhead, and UpVal::v.

Referenced by atomic().

◆ removeentry()

static void removeentry ( Node * n)
static

Definition at line 62 of file lua-5.1.5/src/lgc.c.

62 {
64 if (iscollectable(gkey(n)))
65 setttype(gkey(n), LUA_TDEADKEY); /* dead key; remove it */
66}
#define setttype(obj, tt)
#define LUA_TDEADKEY
#define gkey(n)

References gkey, gval, iscollectable, lua_assert, LUA_TDEADKEY, setttype, and ttisnil.

Referenced by cleartable(), and traversetable().

◆ singlestep()

static l_mem singlestep ( lua_State * L)
static

Definition at line 556 of file lua-5.1.5/src/lgc.c.

556 {
557 global_State *g = G(L);
558 /*lua_checkmemory(L);*/
559 switch (g->gcstate) {
560 case GCSpause: {
561 markroot(L); /* start a new collection */
562 return 0;
563 }
564 case GCSpropagate: {
565 if (g->gray)
566 return propagatemark(g);
567 else { /* no more `gray' objects */
568 atomic(L); /* finish mark phase */
569 return 0;
570 }
571 }
572 case GCSsweepstring: {
573 lu_mem old = g->totalbytes;
574 sweepwholelist(L, &g->strt.hash[g->sweepstrgc++]);
575 if (g->sweepstrgc >= g->strt.size) /* nothing more to sweep? */
576 g->gcstate = GCSsweep; /* end sweep-string phase */
577 lua_assert(old >= g->totalbytes);
578 g->estimate -= old - g->totalbytes;
579 return GCSWEEPCOST;
580 }
581 case GCSsweep: {
582 lu_mem old = g->totalbytes;
583 g->sweepgc = sweeplist(L, g->sweepgc, GCSWEEPMAX);
584 if (*g->sweepgc == NULL) { /* nothing more to sweep? */
585 checkSizes(L);
586 g->gcstate = GCSfinalize; /* end sweep phase */
587 }
588 lua_assert(old >= g->totalbytes);
589 g->estimate -= old - g->totalbytes;
590 return GCSWEEPMAX*GCSWEEPCOST;
591 }
592 case GCSfinalize: {
593 if (g->tmudata) {
594 GCTM(L);
595 if (g->estimate > GCFINALIZECOST)
597 return GCFINALIZECOST;
598 }
599 else {
600 g->gcstate = GCSpause; /* end collection */
601 g->gcdept = 0;
602 return 0;
603 }
604 }
605 default: lua_assert(0); return 0;
606 }
607}
static void atomic(lua_State *L)
#define GCFINALIZECOST
static void checkSizes(lua_State *L)
static GCObject ** sweeplist(lua_State *L, GCObject **p, lu_mem count)
#define GCSWEEPMAX
#define GCSWEEPCOST

References atomic(), checkSizes(), global_State::estimate, G, global_State::gcdept, GCFINALIZECOST, GCSfinalize, GCSpause, GCSpropagate, GCSsweep, GCSsweepstring, global_State::gcstate, GCSWEEPCOST, GCSWEEPMAX, GCTM(), global_State::gray, stringtable::hash, lua_assert, markroot(), NULL, propagatemark(), stringtable::size, global_State::strt, global_State::sweepgc, sweeplist(), global_State::sweepstrgc, sweepwholelist, global_State::tmudata, and global_State::totalbytes.

Referenced by luaC_fullgc(), and luaC_step().

◆ sweeplist()

static GCObject ** sweeplist ( lua_State * L,
GCObject ** p,
lu_mem count )
static

Definition at line 407 of file lua-5.1.5/src/lgc.c.

407 {
408 GCObject *curr;
409 global_State *g = G(L);
410 int deadmask = otherwhite(g);
411 while ((curr = *p) != NULL && count-- > 0) {
412 if (curr->gch.tt == LUA_TTHREAD) /* sweep open upvalues of each thread */
413 sweepwholelist(L, &gco2th(curr)->openupval);
414 if ((curr->gch.marked ^ WHITEBITS) & deadmask) { /* not dead? */
415 lua_assert(!isdead(g, curr) || testbit(curr->gch.marked, FIXEDBIT));
416 makewhite(g, curr); /* make it white (for next cycle) */
417 p = &curr->gch.next;
418 }
419 else { /* must erase `curr' */
420 lua_assert(isdead(g, curr) || deadmask == bitmask(SFIXEDBIT));
421 *p = curr->gch.next;
422 if (curr == g->rootgc) /* is the first element of the list? */
423 g->rootgc = curr->gch.next; /* adjust first */
424 freeobj(L, curr);
425 }
426 }
427 return p;
428}
static void freeobj(lua_State *L, GCObject *o)
#define FIXEDBIT

References bitmask, FIXEDBIT, freeobj(), G, GCObject::gch, gco2th, isdead, lua_assert, LUA_TTHREAD, makewhite, NULL, otherwhite, global_State::rootgc, SFIXEDBIT, sweepwholelist, testbit, and WHITEBITS.

Referenced by singlestep().

◆ traverseclosure()

static void traverseclosure ( global_State * g,
Closure * cl )
static

Definition at line 224 of file lua-5.1.5/src/lgc.c.

224 {
225 markobject(g, cl->c.env);
226 if (cl->c.isC) {
227 int i;
228 for (i=0; i<cl->c.nupvalues; i++) /* mark its upvalues */
229 markvalue(g, &cl->c.upvalue[i]);
230 }
231 else {
232 int i;
233 lua_assert(cl->l.nupvalues == cl->l.p->nups);
234 markobject(g, cl->l.p);
235 for (i=0; i<cl->l.nupvalues; i++) /* mark its upvalues */
236 markobject(g, cl->l.upvals[i]);
237 }
238}
struct Proto * p

References Closure::c, Closure::l, lua_assert, markobject, markvalue, Proto::nups, LClosure::p, LClosure::upvals, and CClosure::upvalue.

Referenced by propagatemark().

◆ traverseproto()

static void traverseproto ( global_State * g,
Proto * f )
static

Definition at line 203 of file lua-5.1.5/src/lgc.c.

203 {
204 int i;
205 if (f->source) stringmark(f->source);
206 for (i=0; i<f->sizek; i++) /* mark literals */
207 markvalue(g, &f->k[i]);
208 for (i=0; i<f->sizeupvalues; i++) { /* mark upvalue names */
209 if (f->upvalues[i])
210 stringmark(f->upvalues[i]);
211 }
212 for (i=0; i<f->sizep; i++) { /* mark nested protos */
213 if (f->p[i])
214 markobject(g, f->p[i]);
215 }
216 for (i=0; i<f->sizelocvars; i++) { /* mark local-variable names */
217 if (f->locvars[i].varname)
219 }
220}
TString * varname
struct LocVar * locvars
struct Proto ** p
TString ** upvalues
TString * source

References Proto::k, Proto::locvars, markobject, markvalue, Proto::p, Proto::sizek, Proto::sizelocvars, Proto::sizep, Proto::sizeupvalues, Proto::source, stringmark, Proto::upvalues, and LocVar::varname.

Referenced by propagatemark().

◆ traversestack()

static void traversestack ( global_State * g,
lua_State * l )
static

Definition at line 256 of file lua-5.1.5/src/lgc.c.

256 {
257 StkId o, lim;
258 CallInfo *ci;
259 markvalue(g, gt(l));
260 lim = l->top;
261 for (ci = l->base_ci; ci <= l->ci; ci++) {
262 lua_assert(ci->top <= l->stack_last);
263 if (lim < ci->top) lim = ci->top;
264 }
265 for (o = l->stack; o < l->top; o++)
266 markvalue(g, o);
267 for (; o <= lim; o++)
268 setnilvalue(o);
269 checkstacksizes(l, lim);
270}
static void checkstacksizes(lua_State *L, StkId max)

References lua_State::base_ci, checkstacksizes(), lua_State::ci, gt, lua_assert, markvalue, setnilvalue, lua_State::stack, lua_State::stack_last, CallInfo::top, and lua_State::top.

Referenced by propagatemark().

◆ traversetable()

static int traversetable ( global_State * g,
Table * h )
static

Definition at line 158 of file lua-5.1.5/src/lgc.c.

158 {
159 int i;
160 int weakkey = 0;
161 int weakvalue = 0;
162 const TValue *mode;
163 if (h->metatable)
164 markobject(g, h->metatable);
165 mode = gfasttm(g, h->metatable, TM_MODE);
166 if (mode && ttisstring(mode)) { /* is there a weak mode? */
167 weakkey = (strchr(svalue(mode), 'k') != NULL);
168 weakvalue = (strchr(svalue(mode), 'v') != NULL);
169 if (weakkey || weakvalue) { /* is really weak? */
170 h->marked &= ~(KEYWEAK | VALUEWEAK); /* clear bits */
171 h->marked |= cast_byte((weakkey << KEYWEAKBIT) |
172 (weakvalue << VALUEWEAKBIT));
173 h->gclist = g->weak; /* must be cleared after GC, ... */
174 g->weak = obj2gco(h); /* ... so put in the appropriate list */
175 }
176 }
177 if (weakkey && weakvalue) return 1;
178 if (!weakvalue) {
179 i = h->sizearray;
180 while (i--)
181 markvalue(g, &h->array[i]);
182 }
183 i = sizenode(h);
184 while (i--) {
185 Node *n = gnode(h, i);
187 if (ttisnil(gval(n)))
188 removeentry(n); /* remove empty entries */
189 else {
190 lua_assert(!ttisnil(gkey(n)));
191 if (!weakkey) markvalue(g, gkey(n));
192 if (!weakvalue) markvalue(g, gval(n));
193 }
194 }
195 return weakkey || weakvalue;
196}
#define KEYWEAK
#define VALUEWEAK
#define svalue(o)
@ TM_MODE
#define gfasttm(g, et, e)

References Table::array, cast_byte, Table::gclist, gfasttm, gkey, gnode, gval, KEYWEAK, KEYWEAKBIT, lua_assert, LUA_TDEADKEY, markobject, markvalue, Table::metatable, NULL, obj2gco, removeentry(), Table::sizearray, sizenode, svalue, TM_MODE, ttisnil, ttisstring, ttype, VALUEWEAK, VALUEWEAKBIT, and global_State::weak.

Referenced by propagatemark().