/src/testdir/build/lua-master/source/lfunc.c
| Line | Count | Source (jump to first uncovered line) | 
| 1 |  | /* | 
| 2 |  | ** $Id: lfunc.c $ | 
| 3 |  | ** Auxiliary functions to manipulate prototypes and closures | 
| 4 |  | ** See Copyright Notice in lua.h | 
| 5 |  | */ | 
| 6 |  |  | 
| 7 |  | #define lfunc_c | 
| 8 |  | #define LUA_CORE | 
| 9 |  |  | 
| 10 |  | #include "lprefix.h" | 
| 11 |  |  | 
| 12 |  |  | 
| 13 |  | #include <stddef.h> | 
| 14 |  |  | 
| 15 |  | #include "lua.h" | 
| 16 |  |  | 
| 17 |  | #include "ldebug.h" | 
| 18 |  | #include "ldo.h" | 
| 19 |  | #include "lfunc.h" | 
| 20 |  | #include "lgc.h" | 
| 21 |  | #include "lmem.h" | 
| 22 |  | #include "lobject.h" | 
| 23 |  | #include "lstate.h" | 
| 24 |  |  | 
| 25 |  |  | 
| 26 |  |  | 
| 27 | 0 | CClosure *luaF_newCclosure (lua_State *L, int nupvals) { | 
| 28 | 0 |   GCObject *o = luaC_newobj(L, LUA_VCCL, sizeCclosure(nupvals)); | 
| 29 | 0 |   CClosure *c = gco2ccl(o); | 
| 30 | 0 |   c->nupvalues = cast_byte(nupvals); | 
| 31 | 0 |   return c; | 
| 32 | 0 | } | 
| 33 |  |  | 
| 34 |  |  | 
| 35 | 1.07M | LClosure *luaF_newLclosure (lua_State *L, int nupvals) { | 
| 36 | 1.07M |   GCObject *o = luaC_newobj(L, LUA_VLCL, sizeLclosure(nupvals)); | 
| 37 | 1.07M |   LClosure *c = gco2lcl(o); | 
| 38 | 0 |   c->p = NULL; | 
| 39 | 1.07M |   c->nupvalues = cast_byte(nupvals); | 
| 40 | 4.24M |   while (nupvals--) c->upvals[nupvals] = NULL; | 
| 41 | 1.07M |   return c; | 
| 42 | 1.07M | } | 
| 43 |  |  | 
| 44 |  |  | 
| 45 |  | /* | 
| 46 |  | ** fill a closure with new closed upvalues | 
| 47 |  | */ | 
| 48 | 162 | void luaF_initupvals (lua_State *L, LClosure *cl) { | 
| 49 | 162 |   int i; | 
| 50 | 324 |   for (i = 0; i < cl->nupvalues; i++) { | 
| 51 | 162 |     GCObject *o = luaC_newobj(L, LUA_VUPVAL, sizeof(UpVal)); | 
| 52 | 162 |     UpVal *uv = gco2upv(o); | 
| 53 | 0 |     uv->v.p = &uv->u.value;  /* make it closed */ | 
| 54 | 162 |     setnilvalue(uv->v.p); | 
| 55 | 162 |     cl->upvals[i] = uv; | 
| 56 | 162 |     luaC_objbarrier(L, cl, uv); | 
| 57 | 162 |   } | 
| 58 | 162 | } | 
| 59 |  |  | 
| 60 |  |  | 
| 61 |  | /* | 
| 62 |  | ** Create a new upvalue at the given level, and link it to the list of | 
| 63 |  | ** open upvalues of 'L' after entry 'prev'. | 
| 64 |  | **/ | 
| 65 | 1.02M | static UpVal *newupval (lua_State *L, StkId level, UpVal **prev) { | 
| 66 | 1.02M |   GCObject *o = luaC_newobj(L, LUA_VUPVAL, sizeof(UpVal)); | 
| 67 | 1.02M |   UpVal *uv = gco2upv(o); | 
| 68 | 0 |   UpVal *next = *prev; | 
| 69 | 1.02M |   uv->v.p = s2v(level);  /* current value lives in the stack */ | 
| 70 | 1.02M |   uv->u.open.next = next;  /* link it to list of open upvalues */ | 
| 71 | 1.02M |   uv->u.open.previous = prev; | 
| 72 | 1.02M |   if (next) | 
| 73 | 1.00M |     next->u.open.previous = &uv->u.open.next; | 
| 74 | 1.02M |   *prev = uv; | 
| 75 | 1.02M |   if (!isintwups(L)) {  /* thread not in list of threads with upvalues? */ | 
| 76 | 19 |     L->twups = G(L)->twups;  /* link it to the list */ | 
| 77 | 19 |     G(L)->twups = L; | 
| 78 | 19 |   } | 
| 79 | 1.02M |   return uv; | 
| 80 | 1.02M | } | 
| 81 |  |  | 
| 82 |  |  | 
| 83 |  | /* | 
| 84 |  | ** Find and reuse, or create if it does not exist, an upvalue | 
| 85 |  | ** at the given level. | 
| 86 |  | */ | 
| 87 | 1.57M | UpVal *luaF_findupval (lua_State *L, StkId level) { | 
| 88 | 1.57M |   UpVal **pp = &L->openupval; | 
| 89 | 1.57M |   UpVal *p; | 
| 90 | 1.57M |   lua_assert(isintwups(L) || L->openupval == NULL); | 
| 91 | 60.2M |   while ((p = *pp) != NULL && uplevel(p) >= level) {  /* search for it */ | 
| 92 | 29.1M |     lua_assert(!isdead(G(L), p)); | 
| 93 | 58.2M |     if (uplevel(p) == level)  /* corresponding upvalue? */ | 
| 94 | 551k |       return p;  /* return it */ | 
| 95 | 28.5M |     pp = &p->u.open.next; | 
| 96 | 28.5M |   } | 
| 97 |  |   /* not found: create a new upvalue after 'pp' */ | 
| 98 | 1.02M |   return newupval(L, level, pp); | 
| 99 | 1.57M | } | 
| 100 |  |  | 
| 101 |  |  | 
| 102 |  | /* | 
| 103 |  | ** Call closing method for object 'obj' with error message 'err'. The | 
| 104 |  | ** boolean 'yy' controls whether the call is yieldable. | 
| 105 |  | ** (This function assumes EXTRA_STACK.) | 
| 106 |  | */ | 
| 107 | 45 | static void callclosemethod (lua_State *L, TValue *obj, TValue *err, int yy) { | 
| 108 | 45 |   StkId top = L->top.p; | 
| 109 | 45 |   const TValue *tm = luaT_gettmbyobj(L, obj, TM_CLOSE); | 
| 110 | 45 |   setobj2s(L, top, tm);  /* will call metamethod... */ | 
| 111 | 45 |   setobj2s(L, top + 1, obj);  /* with 'self' as the 1st argument */ | 
| 112 | 45 |   setobj2s(L, top + 2, err);  /* and error msg. as 2nd argument */ | 
| 113 | 45 |   L->top.p = top + 3;  /* add function and arguments */ | 
| 114 | 45 |   if (yy) | 
| 115 | 0 |     luaD_call(L, top, 0); | 
| 116 | 45 |   else | 
| 117 | 45 |     luaD_callnoyield(L, top, 0); | 
| 118 | 45 | } | 
| 119 |  |  | 
| 120 |  |  | 
| 121 |  | /* | 
| 122 |  | ** Check whether object at given level has a close metamethod and raise | 
| 123 |  | ** an error if not. | 
| 124 |  | */ | 
| 125 | 45 | static void checkclosemth (lua_State *L, StkId level) { | 
| 126 | 45 |   const TValue *tm = luaT_gettmbyobj(L, s2v(level), TM_CLOSE); | 
| 127 | 45 |   if (ttisnil(tm)) {  /* no metamethod? */ | 
| 128 | 0 |     int idx = cast_int(level - L->ci->func.p);  /* variable index */ | 
| 129 | 0 |     const char *vname = luaG_findlocal(L, L->ci, idx, NULL); | 
| 130 | 0 |     if (vname == NULL) vname = "?"; | 
| 131 | 0 |     luaG_runerror(L, "variable '%s' got a non-closable value", vname); | 
| 132 | 0 |   } | 
| 133 | 45 | } | 
| 134 |  |  | 
| 135 |  |  | 
| 136 |  | /* | 
| 137 |  | ** Prepare and call a closing method. | 
| 138 |  | ** If status is CLOSEKTOP, the call to the closing method will be pushed | 
| 139 |  | ** at the top of the stack. Otherwise, values can be pushed right after | 
| 140 |  | ** the 'level' of the upvalue being closed, as everything after that | 
| 141 |  | ** won't be used again. | 
| 142 |  | */ | 
| 143 | 45 | static void prepcallclosemth (lua_State *L, StkId level, int status, int yy) { | 
| 144 | 45 |   TValue *uv = s2v(level);  /* value being closed */ | 
| 145 | 45 |   TValue *errobj; | 
| 146 | 45 |   if (status == CLOSEKTOP) | 
| 147 | 45 |     errobj = &G(L)->nilvalue;  /* error object is nil */ | 
| 148 | 0 |   else {  /* 'luaD_seterrorobj' will set top to level + 2 */ | 
| 149 | 0 |     errobj = s2v(level + 1);  /* error object goes after 'uv' */ | 
| 150 | 0 |     luaD_seterrorobj(L, status, level + 1);  /* set error object */ | 
| 151 | 0 |   } | 
| 152 | 45 |   callclosemethod(L, uv, errobj, yy); | 
| 153 | 45 | } | 
| 154 |  |  | 
| 155 |  |  | 
| 156 |  | /* | 
| 157 |  | ** Maximum value for deltas in 'tbclist', dependent on the type | 
| 158 |  | ** of delta. (This macro assumes that an 'L' is in scope where it | 
| 159 |  | ** is used.) | 
| 160 |  | */ | 
| 161 |  | #define MAXDELTA  \ | 
| 162 | 90 |   ((256ul << ((sizeof(L->stack.p->tbclist.delta) - 1) * 8)) - 1) | 
| 163 |  |  | 
| 164 |  |  | 
| 165 |  | /* | 
| 166 |  | ** Insert a variable in the list of to-be-closed variables. | 
| 167 |  | */ | 
| 168 | 45 | void luaF_newtbcupval (lua_State *L, StkId level) { | 
| 169 | 45 |   lua_assert(level > L->tbclist.p); | 
| 170 | 45 |   if (l_isfalse(s2v(level))) | 
| 171 | 0 |     return;  /* false doesn't need to be closed */ | 
| 172 | 45 |   checkclosemth(L, level);  /* value must have a close method */ | 
| 173 | 45 |   while (cast_uint(level - L->tbclist.p) > MAXDELTA) { | 
| 174 | 0 |     L->tbclist.p += MAXDELTA;  /* create a dummy node at maximum delta */ | 
| 175 | 0 |     L->tbclist.p->tbclist.delta = 0; | 
| 176 | 0 |   } | 
| 177 | 45 |   level->tbclist.delta = cast(unsigned short, level - L->tbclist.p); | 
| 178 | 45 |   L->tbclist.p = level; | 
| 179 | 45 | } | 
| 180 |  |  | 
| 181 |  |  | 
| 182 | 1.02M | void luaF_unlinkupval (UpVal *uv) { | 
| 183 | 1.02M |   lua_assert(upisopen(uv)); | 
| 184 | 1.02M |   *uv->u.open.previous = uv->u.open.next; | 
| 185 | 1.02M |   if (uv->u.open.next) | 
| 186 | 1.01M |     uv->u.open.next->u.open.previous = uv->u.open.previous; | 
| 187 | 1.02M | } | 
| 188 |  |  | 
| 189 |  |  | 
| 190 |  | /* | 
| 191 |  | ** Close all upvalues up to the given stack level. | 
| 192 |  | */ | 
| 193 | 512k | void luaF_closeupval (lua_State *L, StkId level) { | 
| 194 | 512k |   UpVal *uv; | 
| 195 | 512k |   StkId upl;  /* stack index pointed by 'uv' */ | 
| 196 | 1.53M |   while ((uv = L->openupval) != NULL && (upl = uplevel(uv)) >= level) { | 
| 197 | 1.02M |     TValue *slot = &uv->u.value;  /* new position for value */ | 
| 198 | 1.02M |     lua_assert(uplevel(uv) < L->top.p); | 
| 199 | 1.02M |     luaF_unlinkupval(uv);  /* remove upvalue from 'openupval' list */ | 
| 200 | 1.02M |     setobj(L, slot, uv->v.p);  /* move value to upvalue slot */ | 
| 201 | 1.02M |     uv->v.p = slot;  /* now current value lives here */ | 
| 202 | 1.02M |     if (!iswhite(uv)) {  /* neither white nor dead? */ | 
| 203 | 57.6k |       nw2black(uv);  /* closed upvalues cannot be gray */ | 
| 204 | 57.6k |       luaC_barrier(L, uv, slot); | 
| 205 | 57.6k |     } | 
| 206 | 1.02M |   } | 
| 207 | 512k | } | 
| 208 |  |  | 
| 209 |  |  | 
| 210 |  | /* | 
| 211 |  | ** Remove first element from the tbclist plus its dummy nodes. | 
| 212 |  | */ | 
| 213 | 45 | static void poptbclist (lua_State *L) { | 
| 214 | 45 |   StkId tbc = L->tbclist.p; | 
| 215 | 45 |   lua_assert(tbc->tbclist.delta > 0);  /* first element cannot be dummy */ | 
| 216 | 45 |   tbc -= tbc->tbclist.delta; | 
| 217 | 45 |   while (tbc > L->stack.p && tbc->tbclist.delta == 0) | 
| 218 | 0 |     tbc -= MAXDELTA;  /* remove dummy nodes */ | 
| 219 | 45 |   L->tbclist.p = tbc; | 
| 220 | 45 | } | 
| 221 |  |  | 
| 222 |  |  | 
| 223 |  | /* | 
| 224 |  | ** Close all upvalues and to-be-closed variables up to the given stack | 
| 225 |  | ** level. Return restored 'level'. | 
| 226 |  | */ | 
| 227 | 512k | StkId luaF_close (lua_State *L, StkId level, int status, int yy) { | 
| 228 | 512k |   ptrdiff_t levelrel = savestack(L, level); | 
| 229 | 512k |   luaF_closeupval(L, level);  /* first, close the upvalues */ | 
| 230 | 512k |   while (L->tbclist.p >= level) {  /* traverse tbc's down to that level */ | 
| 231 | 45 |     StkId tbc = L->tbclist.p;  /* get variable index */ | 
| 232 | 45 |     poptbclist(L);  /* remove it from list */ | 
| 233 | 45 |     prepcallclosemth(L, tbc, status, yy);  /* close variable */ | 
| 234 | 45 |     level = restorestack(L, levelrel); | 
| 235 | 45 |   } | 
| 236 | 512k |   return level; | 
| 237 | 512k | } | 
| 238 |  |  | 
| 239 |  |  | 
| 240 | 21.7k | Proto *luaF_newproto (lua_State *L) { | 
| 241 | 21.7k |   GCObject *o = luaC_newobj(L, LUA_VPROTO, sizeof(Proto)); | 
| 242 | 21.7k |   Proto *f = gco2p(o); | 
| 243 | 0 |   f->k = NULL; | 
| 244 | 21.7k |   f->sizek = 0; | 
| 245 | 21.7k |   f->p = NULL; | 
| 246 | 21.7k |   f->sizep = 0; | 
| 247 | 21.7k |   f->code = NULL; | 
| 248 | 21.7k |   f->sizecode = 0; | 
| 249 | 21.7k |   f->lineinfo = NULL; | 
| 250 | 21.7k |   f->sizelineinfo = 0; | 
| 251 | 21.7k |   f->abslineinfo = NULL; | 
| 252 | 21.7k |   f->sizeabslineinfo = 0; | 
| 253 | 21.7k |   f->upvalues = NULL; | 
| 254 | 21.7k |   f->sizeupvalues = 0; | 
| 255 | 21.7k |   f->numparams = 0; | 
| 256 | 21.7k |   f->flag = 0; | 
| 257 | 21.7k |   f->maxstacksize = 0; | 
| 258 | 21.7k |   f->locvars = NULL; | 
| 259 | 21.7k |   f->sizelocvars = 0; | 
| 260 | 21.7k |   f->linedefined = 0; | 
| 261 | 21.7k |   f->lastlinedefined = 0; | 
| 262 | 21.7k |   f->source = NULL; | 
| 263 | 21.7k |   return f; | 
| 264 | 21.7k | } | 
| 265 |  |  | 
| 266 |  |  | 
| 267 | 21.7k | void luaF_freeproto (lua_State *L, Proto *f) { | 
| 268 | 21.7k |   if (!(f->flag & PF_FIXED)) { | 
| 269 | 21.7k |     luaM_freearray(L, f->code, f->sizecode); | 
| 270 | 21.7k |     luaM_freearray(L, f->lineinfo, f->sizelineinfo); | 
| 271 | 21.7k |     luaM_freearray(L, f->abslineinfo, f->sizeabslineinfo); | 
| 272 | 21.7k |   } | 
| 273 | 21.7k |   luaM_freearray(L, f->p, f->sizep); | 
| 274 | 21.7k |   luaM_freearray(L, f->k, f->sizek); | 
| 275 | 21.7k |   luaM_freearray(L, f->locvars, f->sizelocvars); | 
| 276 | 21.7k |   luaM_freearray(L, f->upvalues, f->sizeupvalues); | 
| 277 | 21.7k |   luaM_free(L, f); | 
| 278 | 21.7k | } | 
| 279 |  |  | 
| 280 |  |  | 
| 281 |  | /* | 
| 282 |  | ** Look for n-th local variable at line 'line' in function 'func'. | 
| 283 |  | ** Returns NULL if not found. | 
| 284 |  | */ | 
| 285 | 195 | const char *luaF_getlocalname (const Proto *f, int local_number, int pc) { | 
| 286 | 195 |   int i; | 
| 287 | 11.5k |   for (i = 0; i<f->sizelocvars && f->locvars[i].startpc <= pc; i++) { | 
| 288 | 11.3k |     if (pc < f->locvars[i].endpc) {  /* is variable active? */ | 
| 289 | 1.22k |       local_number--; | 
| 290 | 1.22k |       if (local_number == 0) | 
| 291 | 4 |         return getstr(f->locvars[i].varname); | 
| 292 | 1.22k |     } | 
| 293 | 11.3k |   } | 
| 294 | 191 |   return NULL;  /* not found */ | 
| 295 | 195 | } | 
| 296 |  |  |