diff options
Diffstat (limited to 'lib/lua/src/lfunc.c')
| -rw-r--r-- | lib/lua/src/lfunc.c | 107 | 
1 files changed, 77 insertions, 30 deletions
| diff --git a/lib/lua/src/lfunc.c b/lib/lua/src/lfunc.c index 45a60cb..19fdc1a 100644 --- a/lib/lua/src/lfunc.c +++ b/lib/lua/src/lfunc.c @@ -1,13 +1,14 @@  /* -** $Id: lfunc.c,v 1.5 2007-07-25 16:54:32 pixel Exp $ +** $Id: lfunc.c,v 1.6 2007-07-27 10:05:53 pixel Exp $  ** Auxiliary functions to manipulate prototypes and closures  ** See Copyright Notice in lua.h  */ -#include <stdlib.h> +#include <stddef.h>  #define lfunc_c +#define LUA_CORE  #include "lua.h" @@ -18,57 +19,102 @@  #include "lstate.h" -Closure *luaF_newCclosure (lua_State *L, int nelems) { + +Closure *luaF_newCclosure (lua_State *L, int nelems, Table *e) {    Closure *c = cast(Closure *, luaM_malloc(L, sizeCclosure(nelems))); -  luaC_link(L, valtogco(c), LUA_TFUNCTION); +  luaC_link(L, obj2gco(c), LUA_TFUNCTION);    c->c.isC = 1; -  c->c.nupvalues = cast(lu_byte, nelems); +  c->c.env = e; +  c->c.nupvalues = cast_byte(nelems);    return c;  } -Closure *luaF_newLclosure (lua_State *L, int nelems, TObject *e) { +Closure *luaF_newLclosure (lua_State *L, int nelems, Table *e) {    Closure *c = cast(Closure *, luaM_malloc(L, sizeLclosure(nelems))); -  luaC_link(L, valtogco(c), LUA_TFUNCTION); +  luaC_link(L, obj2gco(c), LUA_TFUNCTION);    c->l.isC = 0; -  c->l.g = *e; -  c->l.nupvalues = cast(lu_byte, nelems); +  c->l.env = e; +  c->l.nupvalues = cast_byte(nelems); +  while (nelems--) c->l.upvals[nelems] = NULL;    return c;  } +UpVal *luaF_newupval (lua_State *L) { +  UpVal *uv = luaM_new(L, UpVal); +  luaC_link(L, obj2gco(uv), LUA_TUPVAL); +  uv->v = &uv->u.value; +  setnilvalue(uv->v); +  return uv; +} + +  UpVal *luaF_findupval (lua_State *L, StkId level) { +  global_State *g = G(L);    GCObject **pp = &L->openupval;    UpVal *p; -  UpVal *v; -  while ((p = ngcotouv(*pp)) != NULL && p->v >= level) { -    if (p->v == level) return p; +  UpVal *uv; +  while (*pp != NULL && (p = ngcotouv(*pp))->v >= level) { +    lua_assert(p->v != &p->u.value); +    if (p->v == level) {  /* found a corresponding upvalue? */ +      if (isdead(g, obj2gco(p)))  /* is it dead? */ +        changewhite(obj2gco(p));  /* ressurect it */ +      return p; +    }      pp = &p->next;    } -  v = luaM_new(L, UpVal);  /* not found: create a new one */ -  v->tt = LUA_TUPVAL; -  v->marked = 1;  /* open upvalues should not be collected */ -  v->v = level;  /* current value lives in the stack */ -  v->next = *pp;  /* chain it in the proper position */ -  *pp = valtogco(v); -  return v; +  uv = luaM_new(L, UpVal);  /* not found: create a new one */ +  uv->tt = LUA_TUPVAL; +  uv->marked = luaC_white(g); +  uv->v = level;  /* current value lives in the stack */ +  uv->next = *pp;  /* chain it in the proper position */ +  *pp = obj2gco(uv); +  uv->u.l.prev = &g->uvhead;  /* double link it in `uvhead' list */ +  uv->u.l.next = g->uvhead.u.l.next; +  uv->u.l.next->u.l.prev = uv; +  g->uvhead.u.l.next = uv; +  lua_assert(uv->u.l.next->u.l.prev == uv && uv->u.l.prev->u.l.next == uv); +  return uv; +} + + +static void unlinkupval (UpVal *uv) { +  lua_assert(uv->u.l.next->u.l.prev == uv && uv->u.l.prev->u.l.next == uv); +  uv->u.l.next->u.l.prev = uv->u.l.prev;  /* remove from `uvhead' list */ +  uv->u.l.prev->u.l.next = uv->u.l.next; +} + + +void luaF_freeupval (lua_State *L, UpVal *uv) { +  if (uv->v != &uv->u.value)  /* is it open? */ +    unlinkupval(uv);  /* remove from open list */ +  luaM_free(L, uv);  /* free upvalue */  }  void luaF_close (lua_State *L, StkId level) { -  UpVal *p; -  while ((p = ngcotouv(L->openupval)) != NULL && p->v >= level) { -    setobj(&p->value, p->v);  /* save current value (write barrier) */ -    p->v = &p->value;  /* now current value lives here */ -    L->openupval = p->next;  /* remove from `open' list */ -    luaC_link(L, valtogco(p), LUA_TUPVAL); +  UpVal *uv; +  global_State *g = G(L); +  while (L->openupval != NULL && (uv = ngcotouv(L->openupval))->v >= level) { +    GCObject *o = obj2gco(uv); +    lua_assert(!isblack(o) && uv->v != &uv->u.value); +    L->openupval = uv->next;  /* remove from `open' list */ +    if (isdead(g, o)) +      luaF_freeupval(L, uv);  /* free upvalue */ +    else { +      unlinkupval(uv); +      setobj(L, &uv->u.value, uv->v); +      uv->v = &uv->u.value;  /* now current value lives here */ +      luaC_linkupval(L, uv);  /* link upvalue into `gcroot' list */ +    }    }  }  Proto *luaF_newproto (lua_State *L) {    Proto *f = luaM_new(L, Proto); -  luaC_link(L, valtogco(f), LUA_TPROTO); +  luaC_link(L, obj2gco(f), LUA_TPROTO);    f->k = NULL;    f->sizek = 0;    f->p = NULL; @@ -85,7 +131,8 @@ Proto *luaF_newproto (lua_State *L) {    f->lineinfo = NULL;    f->sizelocvars = 0;    f->locvars = NULL; -  f->lineDefined = 0; +  f->linedefined = 0; +  f->lastlinedefined = 0;    f->source = NULL;    return f;  } @@ -94,18 +141,18 @@ Proto *luaF_newproto (lua_State *L) {  void luaF_freeproto (lua_State *L, Proto *f) {    luaM_freearray(L, f->code, f->sizecode, Instruction);    luaM_freearray(L, f->p, f->sizep, Proto *); -  luaM_freearray(L, f->k, f->sizek, TObject); +  luaM_freearray(L, f->k, f->sizek, TValue);    luaM_freearray(L, f->lineinfo, f->sizelineinfo, int);    luaM_freearray(L, f->locvars, f->sizelocvars, struct LocVar);    luaM_freearray(L, f->upvalues, f->sizeupvalues, TString *); -  luaM_freelem(L, f); +  luaM_free(L, f);  }  void luaF_freeclosure (lua_State *L, Closure *c) {    int size = (c->c.isC) ? sizeCclosure(c->c.nupvalues) :                            sizeLclosure(c->l.nupvalues); -  luaM_free(L, c, size); +  luaM_freemem(L, c, size);  } | 
