/* ** $Id: lgc.c,v 1.3 1997/09/26 16:46:20 roberto Exp roberto $ ** Garbage Collector ** See Copyright Notice in lua.h */ #include "ldo.h" #include "lfunc.h" #include "lgc.h" #include "lmem.h" #include "lobject.h" #include "lstring.h" #include "ltable.h" #include "ltm.h" #include "lua.h" static int markobject (TObject *o); /* ** ======================================================= ** REF mechanism ** ======================================================= */ static struct ref { TObject o; enum {LOCK, HOLD, FREE, COLLECTED} status; } *refArray = NULL; static int refSize = 0; int luaC_ref (TObject *o, int lock) { int ref; if (ttype(o) == LUA_T_NIL) ref = -1; /* special ref for nil */ else { for (ref=0; ref= 0 && ref < refSize) refArray[ref].status = FREE; } TObject* luaC_getref (int ref) { static TObject nul = {LUA_T_NIL, {0}}; if (ref == -1) return &nul; if (ref >= 0 && ref < refSize && (refArray[ref].status == LOCK || refArray[ref].status == HOLD)) return &refArray[ref].o; else return NULL; } static void travlock (void) { int i; for (i=0; ittype) { case LUA_T_STRING: case LUA_T_USERDATA: return o->value.ts->head.marked; case LUA_T_FUNCTION: return o->value.cl->head.marked; case LUA_T_PROTO: return o->value.tf->head.marked; case LUA_T_ARRAY: return o->value.a->head.marked; default: /* nil, number or cfunction */ return 1; } } static void invalidaterefs (void) { int i; for (i=0; ihead.next) { avalue(&t) = l; luaD_gcIM(&t); } } static void strcallIM (TaggedString *l) { TObject o; ttype(&o) = LUA_T_USERDATA; for (; l; l=(TaggedString *)l->head.next) if (l->constindex == -1) { /* is userdata? */ tsvalue(&o) = l; luaD_gcIM(&o); } } static GCnode *listcollect (GCnode *l) { GCnode *frees = NULL; while (l) { GCnode *next = l->next; l->marked = 0; while (next && !next->marked) { l->next = next->next; next->next = frees; frees = next; next = l->next; --luaO_nentities; } l = next; } return frees; } static void strmark (TaggedString *s) { if (!s->head.marked) s->head.marked = 1; } static void protomark (TProtoFunc *f) { if (!f->head.marked) { LocVar *v = f->locvars; int i; f->head.marked = 1; if (f->fileName) strmark(f->fileName); for (i=0; inconsts; i++) markobject(&f->consts[i]); if (v) { for (; v->line != -1; v++) if (v->varname) strmark(v->varname); } } } static void funcmark (Closure *f) { if (!f->head.marked) { int i; f->head.marked = 1; for (i=f->nelems; i>=0; i--) markobject(&f->consts[i]); } } static void hashmark (Hash *h) { if (!h->head.marked) { int i; h->head.marked = 1; for (i=0; iref); markobject(&n->val); } } } } static void globalmark (void) { TaggedString *g; for (g=(TaggedString *)luaS_root.next; g; g=(TaggedString *)g->head.next) if (g->u.globalval.ttype != LUA_T_NIL) { markobject(&g->u.globalval); strmark(g); /* cannot collect non nil global variables */ } } static int markobject (TObject *o) { switch (ttype(o)) { case LUA_T_USERDATA: case LUA_T_STRING: strmark(tsvalue(o)); break; case LUA_T_ARRAY: hashmark(avalue(o)); break; case LUA_T_FUNCTION: case LUA_T_MARK: funcmark(o->value.cl); break; case LUA_T_PROTO: protomark(o->value.tf); break; default: break; /* numbers, cfunctions, etc */ } return 0; } static void call_nilIM (void) { /* signals end of garbage collection */ TObject t; ttype(&t) = LUA_T_NIL; luaD_gcIM(&t); /* end of list */ } #define GARBAGE_BLOCK 150 long luaC_threshold = GARBAGE_BLOCK; static void markall (void) { luaD_travstack(markobject); /* mark stack objects */ globalmark(); /* mark global variable values and names */ travlock(); /* mark locked objects */ luaT_travtagmethods(markobject); /* mark fallbacks */ } long lua_collectgarbage (long limit) { long recovered = luaO_nentities; /* to subtract luaM_new value after gc */ Hash *freetable; TaggedString *freestr; TProtoFunc *freefunc; Closure *freeclos; markall(); invalidaterefs(); freestr = luaS_collector(); freetable = (Hash *)listcollect(&luaH_root); freefunc = (TProtoFunc *)listcollect(&luaF_root); freeclos = (Closure *)listcollect(&luaF_rootcl); recovered = recovered-luaO_nentities; /*printf("==total %ld coletados %ld\n", luaO_nentities+recovered, recovered);*/ luaC_threshold = (limit == 0) ? 2*luaO_nentities : luaO_nentities+limit; hashcallIM(freetable); strcallIM(freestr); call_nilIM(); luaH_free(freetable); luaS_free(freestr); luaF_freeproto(freefunc); luaF_freeclosure(freeclos); luaM_clearbuffer(); return recovered; } void luaC_checkGC (void) { if (luaO_nentities >= luaC_threshold) lua_collectgarbage(0); }