1 /* $NetBSD: lapi.c,v 1.15 2023/06/08 21:12:08 nikita Exp $ */
2
3 /*
4 ** Id: lapi.c
5 ** Lua API
6 ** See Copyright Notice in lua.h
7 */
8
9 #define lapi_c
10 #define LUA_CORE
11
12 #include "lprefix.h"
13
14
15 #ifndef _KERNEL
16 #include <limits.h>
17 #endif /* _KERNEL */
18 #include <stdarg.h>
19 #ifndef _KERNEL
20 #include <string.h>
21 #endif /* _KERNEL */
22
23 #include "lua.h"
24
25 #include "lapi.h"
26 #include "ldebug.h"
27 #include "ldo.h"
28 #include "lfunc.h"
29 #include "lgc.h"
30 #include "lmem.h"
31 #include "lobject.h"
32 #include "lstate.h"
33 #include "lstring.h"
34 #include "ltable.h"
35 #include "ltm.h"
36 #include "lundump.h"
37 #include "lvm.h"
38
39
40
41 const char lua_ident[] =
42 "$LuaVersion: " LUA_COPYRIGHT " $"
43 "$LuaAuthors: " LUA_AUTHORS " $";
44
45
46
47 /*
48 ** Test for a valid index (one that is not the 'nilvalue').
49 ** '!ttisnil(o)' implies 'o != &G(L)->nilvalue', so it is not needed.
50 ** However, it covers the most common cases in a faster way.
51 */
52 #define isvalid(L, o) (!ttisnil(o) || o != &G(L)->nilvalue)
53
54
55 /* test for pseudo index */
56 #define ispseudo(i) ((i) <= LUA_REGISTRYINDEX)
57
58 /* test for upvalue */
59 #define isupvalue(i) ((i) < LUA_REGISTRYINDEX)
60
61
62 /*
63 ** Convert an acceptable index to a pointer to its respective value.
64 ** Non-valid indices return the special nil value 'G(L)->nilvalue'.
65 */
index2value(lua_State * L,int idx)66 static TValue *index2value (lua_State *L, int idx) {
67 CallInfo *ci = L->ci;
68 if (idx > 0) {
69 StkId o = ci->func.p + idx;
70 api_check(L, idx <= ci->top.p - (ci->func.p + 1), "unacceptable index");
71 if (o >= L->top.p) return &G(L)->nilvalue;
72 else return s2v(o);
73 }
74 else if (!ispseudo(idx)) { /* negative index */
75 api_check(L, idx != 0 && -idx <= L->top.p - (ci->func.p + 1),
76 "invalid index");
77 return s2v(L->top.p + idx);
78 }
79 else if (idx == LUA_REGISTRYINDEX)
80 return &G(L)->l_registry;
81 else { /* upvalues */
82 idx = LUA_REGISTRYINDEX - idx;
83 api_check(L, idx <= MAXUPVAL + 1, "upvalue index too large");
84 if (ttisCclosure(s2v(ci->func.p))) { /* C closure? */
85 CClosure *func = clCvalue(s2v(ci->func.p));
86 return (idx <= func->nupvalues) ? &func->upvalue[idx-1]
87 : &G(L)->nilvalue;
88 }
89 else { /* light C function or Lua function (through a hook)?) */
90 api_check(L, ttislcf(s2v(ci->func.p)), "caller not a C function");
91 return &G(L)->nilvalue; /* no upvalues */
92 }
93 }
94 }
95
96
97
98 /*
99 ** Convert a valid actual index (not a pseudo-index) to its address.
100 */
index2stack(lua_State * L,int idx)101 l_sinline StkId index2stack (lua_State *L, int idx) {
102 CallInfo *ci = L->ci;
103 if (idx > 0) {
104 StkId o = ci->func.p + idx;
105 api_check(L, o < L->top.p, "invalid index");
106 return o;
107 }
108 else { /* non-positive index */
109 api_check(L, idx != 0 && -idx <= L->top.p - (ci->func.p + 1),
110 "invalid index");
111 api_check(L, !ispseudo(idx), "invalid index");
112 return L->top.p + idx;
113 }
114 }
115
116
lua_checkstack(lua_State * L,int n)117 LUA_API int lua_checkstack (lua_State *L, int n) {
118 int res;
119 CallInfo *ci;
120 lua_lock(L);
121 ci = L->ci;
122 api_check(L, n >= 0, "negative 'n'");
123 if (L->stack_last.p - L->top.p > n) /* stack large enough? */
124 res = 1; /* yes; check is OK */
125 else /* need to grow stack */
126 res = luaD_growstack(L, n, 0);
127 if (res && ci->top.p < L->top.p + n)
128 ci->top.p = L->top.p + n; /* adjust frame top */
129 lua_unlock(L);
130 return res;
131 }
132
133
lua_xmove(lua_State * from,lua_State * to,int n)134 LUA_API void lua_xmove (lua_State *from, lua_State *to, int n) {
135 int i;
136 if (from == to) return;
137 lua_lock(to);
138 api_checknelems(from, n);
139 api_check(from, G(from) == G(to), "moving among independent states");
140 api_check(from, to->ci->top.p - to->top.p >= n, "stack overflow");
141 from->top.p -= n;
142 for (i = 0; i < n; i++) {
143 setobjs2s(to, to->top.p, from->top.p + i);
144 to->top.p++; /* stack already checked by previous 'api_check' */
145 }
146 lua_unlock(to);
147 }
148
149
lua_atpanic(lua_State * L,lua_CFunction panicf)150 LUA_API lua_CFunction lua_atpanic (lua_State *L, lua_CFunction panicf) {
151 lua_CFunction old;
152 lua_lock(L);
153 old = G(L)->panic;
154 G(L)->panic = panicf;
155 lua_unlock(L);
156 return old;
157 }
158
159
lua_version(lua_State * L)160 LUA_API lua_Number lua_version (lua_State *L) {
161 UNUSED(L);
162 return LUA_VERSION_NUM;
163 }
164
165
166
167 /*
168 ** basic stack manipulation
169 */
170
171
172 /*
173 ** convert an acceptable stack index into an absolute index
174 */
lua_absindex(lua_State * L,int idx)175 LUA_API int lua_absindex (lua_State *L, int idx) {
176 return (idx > 0 || ispseudo(idx))
177 ? idx
178 : cast_int(L->top.p - L->ci->func.p) + idx;
179 }
180
181
lua_gettop(lua_State * L)182 LUA_API int lua_gettop (lua_State *L) {
183 return cast_int(L->top.p - (L->ci->func.p + 1));
184 }
185
186
lua_settop(lua_State * L,int idx)187 LUA_API void lua_settop (lua_State *L, int idx) {
188 CallInfo *ci;
189 StkId func, newtop;
190 ptrdiff_t diff; /* difference for new top */
191 lua_lock(L);
192 ci = L->ci;
193 func = ci->func.p;
194 if (idx >= 0) {
195 api_check(L, idx <= ci->top.p - (func + 1), "new top too large");
196 diff = ((func + 1) + idx) - L->top.p;
197 for (; diff > 0; diff--)
198 setnilvalue(s2v(L->top.p++)); /* clear new slots */
199 }
200 else {
201 api_check(L, -(idx+1) <= (L->top.p - (func + 1)), "invalid new top");
202 diff = idx + 1; /* will "subtract" index (as it is negative) */
203 }
204 api_check(L, L->tbclist.p < L->top.p, "previous pop of an unclosed slot");
205 newtop = L->top.p + diff;
206 if (diff < 0 && L->tbclist.p >= newtop) {
207 lua_assert(hastocloseCfunc(ci->nresults));
208 newtop = luaF_close(L, newtop, CLOSEKTOP, 0);
209 }
210 L->top.p = newtop; /* correct top only after closing any upvalue */
211 lua_unlock(L);
212 }
213
214
lua_closeslot(lua_State * L,int idx)215 LUA_API void lua_closeslot (lua_State *L, int idx) {
216 StkId level;
217 lua_lock(L);
218 level = index2stack(L, idx);
219 api_check(L, hastocloseCfunc(L->ci->nresults) && L->tbclist.p == level,
220 "no variable to close at given level");
221 level = luaF_close(L, level, CLOSEKTOP, 0);
222 setnilvalue(s2v(level));
223 lua_unlock(L);
224 }
225
226
227 /*
228 ** Reverse the stack segment from 'from' to 'to'
229 ** (auxiliary to 'lua_rotate')
230 ** Note that we move(copy) only the value inside the stack.
231 ** (We do not move additional fields that may exist.)
232 */
reverse(lua_State * L,StkId from,StkId to)233 l_sinline void reverse (lua_State *L, StkId from, StkId to) {
234 for (; from < to; from++, to--) {
235 TValue temp;
236 setobj(L, &temp, s2v(from));
237 setobjs2s(L, from, to);
238 setobj2s(L, to, &temp);
239 }
240 }
241
242
243 /*
244 ** Let x = AB, where A is a prefix of length 'n'. Then,
245 ** rotate x n == BA. But BA == (A^r . B^r)^r.
246 */
lua_rotate(lua_State * L,int idx,int n)247 LUA_API void lua_rotate (lua_State *L, int idx, int n) {
248 StkId p, t, m;
249 lua_lock(L);
250 t = L->top.p - 1; /* end of stack segment being rotated */
251 p = index2stack(L, idx); /* start of segment */
252 api_check(L, (n >= 0 ? n : -n) <= (t - p + 1), "invalid 'n'");
253 m = (n >= 0 ? t - n : p - n - 1); /* end of prefix */
254 reverse(L, p, m); /* reverse the prefix with length 'n' */
255 reverse(L, m + 1, t); /* reverse the suffix */
256 reverse(L, p, t); /* reverse the entire segment */
257 lua_unlock(L);
258 }
259
260
lua_copy(lua_State * L,int fromidx,int toidx)261 LUA_API void lua_copy (lua_State *L, int fromidx, int toidx) {
262 TValue *fr, *to;
263 lua_lock(L);
264 fr = index2value(L, fromidx);
265 to = index2value(L, toidx);
266 api_check(L, isvalid(L, to), "invalid index");
267 setobj(L, to, fr);
268 if (isupvalue(toidx)) /* function upvalue? */
269 luaC_barrier(L, clCvalue(s2v(L->ci->func.p)), fr);
270 /* LUA_REGISTRYINDEX does not need gc barrier
271 (collector revisits it before finishing collection) */
272 lua_unlock(L);
273 }
274
275
lua_pushvalue(lua_State * L,int idx)276 LUA_API void lua_pushvalue (lua_State *L, int idx) {
277 lua_lock(L);
278 setobj2s(L, L->top.p, index2value(L, idx));
279 api_incr_top(L);
280 lua_unlock(L);
281 }
282
283
284
285 /*
286 ** access functions (stack -> C)
287 */
288
289
lua_type(lua_State * L,int idx)290 LUA_API int lua_type (lua_State *L, int idx) {
291 const TValue *o = index2value(L, idx);
292 return (isvalid(L, o) ? ttype(o) : LUA_TNONE);
293 }
294
295
lua_typename(lua_State * L,int t)296 LUA_API const char *lua_typename (lua_State *L, int t) {
297 UNUSED(L);
298 api_check(L, LUA_TNONE <= t && t < LUA_NUMTYPES, "invalid type");
299 return ttypename(t);
300 }
301
302
lua_iscfunction(lua_State * L,int idx)303 LUA_API int lua_iscfunction (lua_State *L, int idx) {
304 const TValue *o = index2value(L, idx);
305 return (ttislcf(o) || (ttisCclosure(o)));
306 }
307
308
lua_isinteger(lua_State * L,int idx)309 LUA_API int lua_isinteger (lua_State *L, int idx) {
310 const TValue *o = index2value(L, idx);
311 return ttisinteger(o);
312 }
313
314
lua_isnumber(lua_State * L,int idx)315 LUA_API int lua_isnumber (lua_State *L, int idx) {
316 lua_Number n;
317 const TValue *o = index2value(L, idx);
318 return tonumber(o, &n);
319 }
320
321
lua_isstring(lua_State * L,int idx)322 LUA_API int lua_isstring (lua_State *L, int idx) {
323 const TValue *o = index2value(L, idx);
324 return (ttisstring(o) || cvt2str(o));
325 }
326
327
lua_isuserdata(lua_State * L,int idx)328 LUA_API int lua_isuserdata (lua_State *L, int idx) {
329 const TValue *o = index2value(L, idx);
330 return (ttisfulluserdata(o) || ttislightuserdata(o));
331 }
332
333
lua_rawequal(lua_State * L,int index1,int index2)334 LUA_API int lua_rawequal (lua_State *L, int index1, int index2) {
335 const TValue *o1 = index2value(L, index1);
336 const TValue *o2 = index2value(L, index2);
337 return (isvalid(L, o1) && isvalid(L, o2)) ? luaV_rawequalobj(o1, o2) : 0;
338 }
339
340
lua_arith(lua_State * L,int op)341 LUA_API void lua_arith (lua_State *L, int op) {
342 lua_lock(L);
343 if (op != LUA_OPUNM && op != LUA_OPBNOT)
344 api_checknelems(L, 2); /* all other operations expect two operands */
345 else { /* for unary operations, add fake 2nd operand */
346 api_checknelems(L, 1);
347 setobjs2s(L, L->top.p, L->top.p - 1);
348 api_incr_top(L);
349 }
350 /* first operand at top - 2, second at top - 1; result go to top - 2 */
351 luaO_arith(L, op, s2v(L->top.p - 2), s2v(L->top.p - 1), L->top.p - 2);
352 L->top.p--; /* remove second operand */
353 lua_unlock(L);
354 }
355
356
lua_compare(lua_State * L,int index1,int index2,int op)357 LUA_API int lua_compare (lua_State *L, int index1, int index2, int op) {
358 const TValue *o1;
359 const TValue *o2;
360 int i = 0;
361 lua_lock(L); /* may call tag method */
362 o1 = index2value(L, index1);
363 o2 = index2value(L, index2);
364 if (isvalid(L, o1) && isvalid(L, o2)) {
365 switch (op) {
366 case LUA_OPEQ: i = luaV_equalobj(L, o1, o2); break;
367 case LUA_OPLT: i = luaV_lessthan(L, o1, o2); break;
368 case LUA_OPLE: i = luaV_lessequal(L, o1, o2); break;
369 default: api_check(L, 0, "invalid option");
370 }
371 }
372 lua_unlock(L);
373 return i;
374 }
375
376
lua_stringtonumber(lua_State * L,const char * s)377 LUA_API size_t lua_stringtonumber (lua_State *L, const char *s) {
378 size_t sz = luaO_str2num(s, s2v(L->top.p));
379 if (sz != 0)
380 api_incr_top(L);
381 return sz;
382 }
383
384
385 #ifndef _KERNEL
lua_tonumberx(lua_State * L,int idx,int * pisnum)386 LUA_API lua_Number lua_tonumberx (lua_State *L, int idx, int *pisnum) {
387 lua_Number n = 0;
388 const TValue *o = index2value(L, idx);
389 int isnum = tonumber(o, &n);
390 if (pisnum)
391 *pisnum = isnum;
392 return n;
393 }
394 #endif /* _KERNEL */
395
396
lua_tointegerx(lua_State * L,int idx,int * pisnum)397 LUA_API lua_Integer lua_tointegerx (lua_State *L, int idx, int *pisnum) {
398 lua_Integer res = 0;
399 const TValue *o = index2value(L, idx);
400 int isnum = tointeger(o, &res);
401 if (pisnum)
402 *pisnum = isnum;
403 return res;
404 }
405
406
lua_toboolean(lua_State * L,int idx)407 LUA_API int lua_toboolean (lua_State *L, int idx) {
408 const TValue *o = index2value(L, idx);
409 return !l_isfalse(o);
410 }
411
412
lua_tolstring(lua_State * L,int idx,size_t * len)413 LUA_API const char *lua_tolstring (lua_State *L, int idx, size_t *len) {
414 TValue *o;
415 lua_lock(L);
416 o = index2value(L, idx);
417 if (!ttisstring(o)) {
418 if (!cvt2str(o)) { /* not convertible? */
419 if (len != NULL) *len = 0;
420 lua_unlock(L);
421 return NULL;
422 }
423 luaO_tostring(L, o);
424 luaC_checkGC(L);
425 o = index2value(L, idx); /* previous call may reallocate the stack */
426 }
427 if (len != NULL)
428 *len = vslen(o);
429 lua_unlock(L);
430 return svalue(o);
431 }
432
433
lua_rawlen(lua_State * L,int idx)434 LUA_API lua_Unsigned lua_rawlen (lua_State *L, int idx) {
435 const TValue *o = index2value(L, idx);
436 switch (ttypetag(o)) {
437 case LUA_VSHRSTR: return tsvalue(o)->shrlen;
438 case LUA_VLNGSTR: return tsvalue(o)->u.lnglen;
439 case LUA_VUSERDATA: return uvalue(o)->len;
440 case LUA_VTABLE: return luaH_getn(hvalue(o));
441 default: return 0;
442 }
443 }
444
445
lua_tocfunction(lua_State * L,int idx)446 LUA_API lua_CFunction lua_tocfunction (lua_State *L, int idx) {
447 const TValue *o = index2value(L, idx);
448 if (ttislcf(o)) return fvalue(o);
449 else if (ttisCclosure(o))
450 return clCvalue(o)->f;
451 else return NULL; /* not a C function */
452 }
453
454
touserdata(const TValue * o)455 l_sinline void *touserdata (const TValue *o) {
456 switch (ttype(o)) {
457 case LUA_TUSERDATA: return getudatamem(uvalue(o));
458 case LUA_TLIGHTUSERDATA: return pvalue(o);
459 default: return NULL;
460 }
461 }
462
463
lua_touserdata(lua_State * L,int idx)464 LUA_API void *lua_touserdata (lua_State *L, int idx) {
465 const TValue *o = index2value(L, idx);
466 return touserdata(o);
467 }
468
469
lua_tothread(lua_State * L,int idx)470 LUA_API lua_State *lua_tothread (lua_State *L, int idx) {
471 const TValue *o = index2value(L, idx);
472 return (!ttisthread(o)) ? NULL : thvalue(o);
473 }
474
475
476 /*
477 ** Returns a pointer to the internal representation of an object.
478 ** Note that ANSI C does not allow the conversion of a pointer to
479 ** function to a 'void*', so the conversion here goes through
480 ** a 'size_t'. (As the returned pointer is only informative, this
481 ** conversion should not be a problem.)
482 */
lua_topointer(lua_State * L,int idx)483 LUA_API const void *lua_topointer (lua_State *L, int idx) {
484 const TValue *o = index2value(L, idx);
485 switch (ttypetag(o)) {
486 case LUA_VLCF: return cast_voidp(cast_sizet(fvalue(o)));
487 case LUA_VUSERDATA: case LUA_VLIGHTUSERDATA:
488 return touserdata(o);
489 default: {
490 if (iscollectable(o))
491 return gcvalue(o);
492 else
493 return NULL;
494 }
495 }
496 }
497
498
499
500 /*
501 ** push functions (C -> stack)
502 */
503
504
lua_pushnil(lua_State * L)505 LUA_API void lua_pushnil (lua_State *L) {
506 lua_lock(L);
507 setnilvalue(s2v(L->top.p));
508 api_incr_top(L);
509 lua_unlock(L);
510 }
511
512
513 #ifndef _KERNEL
lua_pushnumber(lua_State * L,lua_Number n)514 LUA_API void lua_pushnumber (lua_State *L, lua_Number n) {
515 lua_lock(L);
516 setfltvalue(s2v(L->top.p), n);
517 api_incr_top(L);
518 lua_unlock(L);
519 }
520 #endif /* _KERNEL */
521
522
lua_pushinteger(lua_State * L,lua_Integer n)523 LUA_API void lua_pushinteger (lua_State *L, lua_Integer n) {
524 lua_lock(L);
525 setivalue(s2v(L->top.p), n);
526 api_incr_top(L);
527 lua_unlock(L);
528 }
529
530
531 /*
532 ** Pushes on the stack a string with given length. Avoid using 's' when
533 ** 'len' == 0 (as 's' can be NULL in that case), due to later use of
534 ** 'memcmp' and 'memcpy'.
535 */
lua_pushlstring(lua_State * L,const char * s,size_t len)536 LUA_API const char *lua_pushlstring (lua_State *L, const char *s, size_t len) {
537 TString *ts;
538 lua_lock(L);
539 ts = (len == 0) ? luaS_new(L, "") : luaS_newlstr(L, s, len);
540 setsvalue2s(L, L->top.p, ts);
541 api_incr_top(L);
542 luaC_checkGC(L);
543 lua_unlock(L);
544 return getstr(ts);
545 }
546
547
lua_pushstring(lua_State * L,const char * s)548 LUA_API const char *lua_pushstring (lua_State *L, const char *s) {
549 lua_lock(L);
550 if (s == NULL)
551 setnilvalue(s2v(L->top.p));
552 else {
553 TString *ts;
554 ts = luaS_new(L, s);
555 setsvalue2s(L, L->top.p, ts);
556 s = getstr(ts); /* internal copy's address */
557 }
558 api_incr_top(L);
559 luaC_checkGC(L);
560 lua_unlock(L);
561 return s;
562 }
563
564
lua_pushvfstring(lua_State * L,const char * fmt,va_list argp)565 LUA_API const char *lua_pushvfstring (lua_State *L, const char *fmt,
566 va_list argp) {
567 const char *ret;
568 lua_lock(L);
569 ret = luaO_pushvfstring(L, fmt, argp);
570 luaC_checkGC(L);
571 lua_unlock(L);
572 return ret;
573 }
574
575
lua_pushfstring(lua_State * L,const char * fmt,...)576 LUA_API const char *lua_pushfstring (lua_State *L, const char *fmt, ...) {
577 const char *ret;
578 va_list argp;
579 lua_lock(L);
580 va_start(argp, fmt);
581 ret = luaO_pushvfstring(L, fmt, argp);
582 va_end(argp);
583 luaC_checkGC(L);
584 lua_unlock(L);
585 return ret;
586 }
587
588
lua_pushcclosure(lua_State * L,lua_CFunction fn,int n)589 LUA_API void lua_pushcclosure (lua_State *L, lua_CFunction fn, int n) {
590 lua_lock(L);
591 if (n == 0) {
592 setfvalue(s2v(L->top.p), fn);
593 api_incr_top(L);
594 }
595 else {
596 CClosure *cl;
597 api_checknelems(L, n);
598 api_check(L, n <= MAXUPVAL, "upvalue index too large");
599 cl = luaF_newCclosure(L, n);
600 cl->f = fn;
601 L->top.p -= n;
602 while (n--) {
603 setobj2n(L, &cl->upvalue[n], s2v(L->top.p + n));
604 /* does not need barrier because closure is white */
605 lua_assert(iswhite(cl));
606 }
607 setclCvalue(L, s2v(L->top.p), cl);
608 api_incr_top(L);
609 luaC_checkGC(L);
610 }
611 lua_unlock(L);
612 }
613
614
lua_pushboolean(lua_State * L,int b)615 LUA_API void lua_pushboolean (lua_State *L, int b) {
616 lua_lock(L);
617 if (b)
618 setbtvalue(s2v(L->top.p));
619 else
620 setbfvalue(s2v(L->top.p));
621 api_incr_top(L);
622 lua_unlock(L);
623 }
624
625
lua_pushlightuserdata(lua_State * L,void * p)626 LUA_API void lua_pushlightuserdata (lua_State *L, void *p) {
627 lua_lock(L);
628 setpvalue(s2v(L->top.p), p);
629 api_incr_top(L);
630 lua_unlock(L);
631 }
632
633
lua_pushthread(lua_State * L)634 LUA_API int lua_pushthread (lua_State *L) {
635 lua_lock(L);
636 setthvalue(L, s2v(L->top.p), L);
637 api_incr_top(L);
638 lua_unlock(L);
639 return (G(L)->mainthread == L);
640 }
641
642
643
644 /*
645 ** get functions (Lua -> stack)
646 */
647
648
auxgetstr(lua_State * L,const TValue * t,const char * k)649 l_sinline int auxgetstr (lua_State *L, const TValue *t, const char *k) {
650 const TValue *slot;
651 TString *str = luaS_new(L, k);
652 if (luaV_fastget(L, t, str, slot, luaH_getstr)) {
653 setobj2s(L, L->top.p, slot);
654 api_incr_top(L);
655 }
656 else {
657 setsvalue2s(L, L->top.p, str);
658 api_incr_top(L);
659 luaV_finishget(L, t, s2v(L->top.p - 1), L->top.p - 1, slot);
660 }
661 lua_unlock(L);
662 return ttype(s2v(L->top.p - 1));
663 }
664
665
666 /*
667 ** Get the global table in the registry. Since all predefined
668 ** indices in the registry were inserted right when the registry
669 ** was created and never removed, they must always be in the array
670 ** part of the registry.
671 */
672 #define getGtable(L) \
673 (&hvalue(&G(L)->l_registry)->array[LUA_RIDX_GLOBALS - 1])
674
675
lua_getglobal(lua_State * L,const char * name)676 LUA_API int lua_getglobal (lua_State *L, const char *name) {
677 const TValue *G;
678 lua_lock(L);
679 G = getGtable(L);
680 return auxgetstr(L, G, name);
681 }
682
683
lua_gettable(lua_State * L,int idx)684 LUA_API int lua_gettable (lua_State *L, int idx) {
685 const TValue *slot;
686 TValue *t;
687 lua_lock(L);
688 t = index2value(L, idx);
689 if (luaV_fastget(L, t, s2v(L->top.p - 1), slot, luaH_get)) {
690 setobj2s(L, L->top.p - 1, slot);
691 }
692 else
693 luaV_finishget(L, t, s2v(L->top.p - 1), L->top.p - 1, slot);
694 lua_unlock(L);
695 return ttype(s2v(L->top.p - 1));
696 }
697
698
lua_getfield(lua_State * L,int idx,const char * k)699 LUA_API int lua_getfield (lua_State *L, int idx, const char *k) {
700 lua_lock(L);
701 return auxgetstr(L, index2value(L, idx), k);
702 }
703
704
lua_geti(lua_State * L,int idx,lua_Integer n)705 LUA_API int lua_geti (lua_State *L, int idx, lua_Integer n) {
706 TValue *t;
707 const TValue *slot;
708 lua_lock(L);
709 t = index2value(L, idx);
710 if (luaV_fastgeti(L, t, n, slot)) {
711 setobj2s(L, L->top.p, slot);
712 }
713 else {
714 TValue aux;
715 setivalue(&aux, n);
716 luaV_finishget(L, t, &aux, L->top.p, slot);
717 }
718 api_incr_top(L);
719 lua_unlock(L);
720 return ttype(s2v(L->top.p - 1));
721 }
722
723
finishrawget(lua_State * L,const TValue * val)724 l_sinline int finishrawget (lua_State *L, const TValue *val) {
725 if (isempty(val)) /* avoid copying empty items to the stack */
726 setnilvalue(s2v(L->top.p));
727 else
728 setobj2s(L, L->top.p, val);
729 api_incr_top(L);
730 lua_unlock(L);
731 return ttype(s2v(L->top.p - 1));
732 }
733
734
gettable(lua_State * L,int idx)735 static Table *gettable (lua_State *L, int idx) {
736 TValue *t = index2value(L, idx);
737 api_check(L, ttistable(t), "table expected");
738 return hvalue(t);
739 }
740
741
lua_rawget(lua_State * L,int idx)742 LUA_API int lua_rawget (lua_State *L, int idx) {
743 Table *t;
744 const TValue *val;
745 lua_lock(L);
746 api_checknelems(L, 1);
747 t = gettable(L, idx);
748 val = luaH_get(t, s2v(L->top.p - 1));
749 L->top.p--; /* remove key */
750 return finishrawget(L, val);
751 }
752
753
lua_rawgeti(lua_State * L,int idx,lua_Integer n)754 LUA_API int lua_rawgeti (lua_State *L, int idx, lua_Integer n) {
755 Table *t;
756 lua_lock(L);
757 t = gettable(L, idx);
758 return finishrawget(L, luaH_getint(t, n));
759 }
760
761
lua_rawgetp(lua_State * L,int idx,const void * p)762 LUA_API int lua_rawgetp (lua_State *L, int idx, const void *p) {
763 Table *t;
764 TValue k;
765 lua_lock(L);
766 t = gettable(L, idx);
767 setpvalue(&k, cast_voidp(p));
768 return finishrawget(L, luaH_get(t, &k));
769 }
770
771
lua_createtable(lua_State * L,int narray,int nrec)772 LUA_API void lua_createtable (lua_State *L, int narray, int nrec) {
773 Table *t;
774 lua_lock(L);
775 t = luaH_new(L);
776 sethvalue2s(L, L->top.p, t);
777 api_incr_top(L);
778 if (narray > 0 || nrec > 0)
779 luaH_resize(L, t, narray, nrec);
780 luaC_checkGC(L);
781 lua_unlock(L);
782 }
783
784
lua_getmetatable(lua_State * L,int objindex)785 LUA_API int lua_getmetatable (lua_State *L, int objindex) {
786 const TValue *obj;
787 Table *mt;
788 int res = 0;
789 lua_lock(L);
790 obj = index2value(L, objindex);
791 switch (ttype(obj)) {
792 case LUA_TTABLE:
793 mt = hvalue(obj)->metatable;
794 break;
795 case LUA_TUSERDATA:
796 mt = uvalue(obj)->metatable;
797 break;
798 default:
799 mt = G(L)->mt[ttype(obj)];
800 break;
801 }
802 if (mt != NULL) {
803 sethvalue2s(L, L->top.p, mt);
804 api_incr_top(L);
805 res = 1;
806 }
807 lua_unlock(L);
808 return res;
809 }
810
811
lua_getiuservalue(lua_State * L,int idx,int n)812 LUA_API int lua_getiuservalue (lua_State *L, int idx, int n) {
813 TValue *o;
814 int t;
815 lua_lock(L);
816 o = index2value(L, idx);
817 api_check(L, ttisfulluserdata(o), "full userdata expected");
818 if (n <= 0 || n > uvalue(o)->nuvalue) {
819 setnilvalue(s2v(L->top.p));
820 t = LUA_TNONE;
821 }
822 else {
823 setobj2s(L, L->top.p, &uvalue(o)->uv[n - 1].uv);
824 t = ttype(s2v(L->top.p));
825 }
826 api_incr_top(L);
827 lua_unlock(L);
828 return t;
829 }
830
831
832 /*
833 ** set functions (stack -> Lua)
834 */
835
836 /*
837 ** t[k] = value at the top of the stack (where 'k' is a string)
838 */
auxsetstr(lua_State * L,const TValue * t,const char * k)839 static void auxsetstr (lua_State *L, const TValue *t, const char *k) {
840 const TValue *slot;
841 TString *str = luaS_new(L, k);
842 api_checknelems(L, 1);
843 if (luaV_fastget(L, t, str, slot, luaH_getstr)) {
844 luaV_finishfastset(L, t, slot, s2v(L->top.p - 1));
845 L->top.p--; /* pop value */
846 }
847 else {
848 setsvalue2s(L, L->top.p, str); /* push 'str' (to make it a TValue) */
849 api_incr_top(L);
850 luaV_finishset(L, t, s2v(L->top.p - 1), s2v(L->top.p - 2), slot);
851 L->top.p -= 2; /* pop value and key */
852 }
853 lua_unlock(L); /* lock done by caller */
854 }
855
856
lua_setglobal(lua_State * L,const char * name)857 LUA_API void lua_setglobal (lua_State *L, const char *name) {
858 const TValue *G;
859 lua_lock(L); /* unlock done in 'auxsetstr' */
860 G = getGtable(L);
861 auxsetstr(L, G, name);
862 }
863
864
lua_settable(lua_State * L,int idx)865 LUA_API void lua_settable (lua_State *L, int idx) {
866 TValue *t;
867 const TValue *slot;
868 lua_lock(L);
869 api_checknelems(L, 2);
870 t = index2value(L, idx);
871 if (luaV_fastget(L, t, s2v(L->top.p - 2), slot, luaH_get)) {
872 luaV_finishfastset(L, t, slot, s2v(L->top.p - 1));
873 }
874 else
875 luaV_finishset(L, t, s2v(L->top.p - 2), s2v(L->top.p - 1), slot);
876 L->top.p -= 2; /* pop index and value */
877 lua_unlock(L);
878 }
879
880
lua_setfield(lua_State * L,int idx,const char * k)881 LUA_API void lua_setfield (lua_State *L, int idx, const char *k) {
882 lua_lock(L); /* unlock done in 'auxsetstr' */
883 auxsetstr(L, index2value(L, idx), k);
884 }
885
886
lua_seti(lua_State * L,int idx,lua_Integer n)887 LUA_API void lua_seti (lua_State *L, int idx, lua_Integer n) {
888 TValue *t;
889 const TValue *slot;
890 lua_lock(L);
891 api_checknelems(L, 1);
892 t = index2value(L, idx);
893 if (luaV_fastgeti(L, t, n, slot)) {
894 luaV_finishfastset(L, t, slot, s2v(L->top.p - 1));
895 }
896 else {
897 TValue aux;
898 setivalue(&aux, n);
899 luaV_finishset(L, t, &aux, s2v(L->top.p - 1), slot);
900 }
901 L->top.p--; /* pop value */
902 lua_unlock(L);
903 }
904
905
aux_rawset(lua_State * L,int idx,TValue * key,int n)906 static void aux_rawset (lua_State *L, int idx, TValue *key, int n) {
907 Table *t;
908 lua_lock(L);
909 api_checknelems(L, n);
910 t = gettable(L, idx);
911 luaH_set(L, t, key, s2v(L->top.p - 1));
912 invalidateTMcache(t);
913 luaC_barrierback(L, obj2gco(t), s2v(L->top.p - 1));
914 L->top.p -= n;
915 lua_unlock(L);
916 }
917
918
lua_rawset(lua_State * L,int idx)919 LUA_API void lua_rawset (lua_State *L, int idx) {
920 aux_rawset(L, idx, s2v(L->top.p - 2), 2);
921 }
922
923
lua_rawsetp(lua_State * L,int idx,const void * p)924 LUA_API void lua_rawsetp (lua_State *L, int idx, const void *p) {
925 TValue k;
926 setpvalue(&k, cast_voidp(p));
927 aux_rawset(L, idx, &k, 1);
928 }
929
930
lua_rawseti(lua_State * L,int idx,lua_Integer n)931 LUA_API void lua_rawseti (lua_State *L, int idx, lua_Integer n) {
932 Table *t;
933 lua_lock(L);
934 api_checknelems(L, 1);
935 t = gettable(L, idx);
936 luaH_setint(L, t, n, s2v(L->top.p - 1));
937 luaC_barrierback(L, obj2gco(t), s2v(L->top.p - 1));
938 L->top.p--;
939 lua_unlock(L);
940 }
941
942
lua_setmetatable(lua_State * L,int objindex)943 LUA_API int lua_setmetatable (lua_State *L, int objindex) {
944 TValue *obj;
945 Table *mt;
946 lua_lock(L);
947 api_checknelems(L, 1);
948 obj = index2value(L, objindex);
949 if (ttisnil(s2v(L->top.p - 1)))
950 mt = NULL;
951 else {
952 api_check(L, ttistable(s2v(L->top.p - 1)), "table expected");
953 mt = hvalue(s2v(L->top.p - 1));
954 }
955 switch (ttype(obj)) {
956 case LUA_TTABLE: {
957 hvalue(obj)->metatable = mt;
958 if (mt) {
959 luaC_objbarrier(L, gcvalue(obj), mt);
960 luaC_checkfinalizer(L, gcvalue(obj), mt);
961 }
962 break;
963 }
964 case LUA_TUSERDATA: {
965 uvalue(obj)->metatable = mt;
966 if (mt) {
967 luaC_objbarrier(L, uvalue(obj), mt);
968 luaC_checkfinalizer(L, gcvalue(obj), mt);
969 }
970 break;
971 }
972 default: {
973 G(L)->mt[ttype(obj)] = mt;
974 break;
975 }
976 }
977 L->top.p--;
978 lua_unlock(L);
979 return 1;
980 }
981
982
lua_setiuservalue(lua_State * L,int idx,int n)983 LUA_API int lua_setiuservalue (lua_State *L, int idx, int n) {
984 TValue *o;
985 int res;
986 lua_lock(L);
987 api_checknelems(L, 1);
988 o = index2value(L, idx);
989 api_check(L, ttisfulluserdata(o), "full userdata expected");
990 if (!(cast_uint(n) - 1u < cast_uint(uvalue(o)->nuvalue)))
991 res = 0; /* 'n' not in [1, uvalue(o)->nuvalue] */
992 else {
993 setobj(L, &uvalue(o)->uv[n - 1].uv, s2v(L->top.p - 1));
994 luaC_barrierback(L, gcvalue(o), s2v(L->top.p - 1));
995 res = 1;
996 }
997 L->top.p--;
998 lua_unlock(L);
999 return res;
1000 }
1001
1002
1003 /*
1004 ** 'load' and 'call' functions (run Lua code)
1005 */
1006
1007
1008 #define checkresults(L,na,nr) \
1009 api_check(L, (nr) == LUA_MULTRET \
1010 || (L->ci->top.p - L->top.p >= (nr) - (na)), \
1011 "results from function overflow current stack size")
1012
1013
lua_callk(lua_State * L,int nargs,int nresults,lua_KContext ctx,lua_KFunction k)1014 LUA_API void lua_callk (lua_State *L, int nargs, int nresults,
1015 lua_KContext ctx, lua_KFunction k) {
1016 StkId func;
1017 lua_lock(L);
1018 api_check(L, k == NULL || !isLua(L->ci),
1019 "cannot use continuations inside hooks");
1020 api_checknelems(L, nargs+1);
1021 api_check(L, L->status == LUA_OK, "cannot do calls on non-normal thread");
1022 checkresults(L, nargs, nresults);
1023 func = L->top.p - (nargs+1);
1024 if (k != NULL && yieldable(L)) { /* need to prepare continuation? */
1025 L->ci->u.c.k = k; /* save continuation */
1026 L->ci->u.c.ctx = ctx; /* save context */
1027 luaD_call(L, func, nresults); /* do the call */
1028 }
1029 else /* no continuation or no yieldable */
1030 luaD_callnoyield(L, func, nresults); /* just do the call */
1031 adjustresults(L, nresults);
1032 lua_unlock(L);
1033 }
1034
1035
1036
1037 /*
1038 ** Execute a protected call.
1039 */
1040 struct CallS { /* data to 'f_call' */
1041 StkId func;
1042 int nresults;
1043 };
1044
1045
f_call(lua_State * L,void * ud)1046 static void f_call (lua_State *L, void *ud) {
1047 struct CallS *c = cast(struct CallS *, ud);
1048 luaD_callnoyield(L, c->func, c->nresults);
1049 }
1050
1051
1052
lua_pcallk(lua_State * L,int nargs,int nresults,int errfunc,lua_KContext ctx,lua_KFunction k)1053 LUA_API int lua_pcallk (lua_State *L, int nargs, int nresults, int errfunc,
1054 lua_KContext ctx, lua_KFunction k) {
1055 struct CallS c;
1056 int status;
1057 ptrdiff_t func;
1058 lua_lock(L);
1059 api_check(L, k == NULL || !isLua(L->ci),
1060 "cannot use continuations inside hooks");
1061 api_checknelems(L, nargs+1);
1062 api_check(L, L->status == LUA_OK, "cannot do calls on non-normal thread");
1063 checkresults(L, nargs, nresults);
1064 if (errfunc == 0)
1065 func = 0;
1066 else {
1067 StkId o = index2stack(L, errfunc);
1068 api_check(L, ttisfunction(s2v(o)), "error handler must be a function");
1069 func = savestack(L, o);
1070 }
1071 c.func = L->top.p - (nargs+1); /* function to be called */
1072 if (k == NULL || !yieldable(L)) { /* no continuation or no yieldable? */
1073 c.nresults = nresults; /* do a 'conventional' protected call */
1074 status = luaD_pcall(L, f_call, &c, savestack(L, c.func), func);
1075 }
1076 else { /* prepare continuation (call is already protected by 'resume') */
1077 CallInfo *ci = L->ci;
1078 ci->u.c.k = k; /* save continuation */
1079 ci->u.c.ctx = ctx; /* save context */
1080 /* save information for error recovery */
1081 ci->u2.funcidx = cast_int(savestack(L, c.func));
1082 ci->u.c.old_errfunc = L->errfunc;
1083 L->errfunc = func;
1084 setoah(ci->callstatus, L->allowhook); /* save value of 'allowhook' */
1085 ci->callstatus |= CIST_YPCALL; /* function can do error recovery */
1086 luaD_call(L, c.func, nresults); /* do the call */
1087 ci->callstatus &= ~CIST_YPCALL;
1088 L->errfunc = ci->u.c.old_errfunc;
1089 status = LUA_OK; /* if it is here, there were no errors */
1090 }
1091 adjustresults(L, nresults);
1092 lua_unlock(L);
1093 return status;
1094 }
1095
1096
lua_load(lua_State * L,lua_Reader reader,void * data,const char * chunkname,const char * mode)1097 LUA_API int lua_load (lua_State *L, lua_Reader reader, void *data,
1098 const char *chunkname, const char *mode) {
1099 ZIO z;
1100 int status;
1101 lua_lock(L);
1102 if (!chunkname) chunkname = "?";
1103 luaZ_init(L, &z, reader, data);
1104 status = luaD_protectedparser(L, &z, chunkname, mode);
1105 if (status == LUA_OK) { /* no errors? */
1106 LClosure *f = clLvalue(s2v(L->top.p - 1)); /* get new function */
1107 if (f->nupvalues >= 1) { /* does it have an upvalue? */
1108 /* get global table from registry */
1109 const TValue *gt = getGtable(L);
1110 /* set global table as 1st upvalue of 'f' (may be LUA_ENV) */
1111 setobj(L, f->upvals[0]->v.p, gt);
1112 luaC_barrier(L, f->upvals[0], gt);
1113 }
1114 }
1115 lua_unlock(L);
1116 return status;
1117 }
1118
1119
lua_dump(lua_State * L,lua_Writer writer,void * data,int strip)1120 LUA_API int lua_dump (lua_State *L, lua_Writer writer, void *data, int strip) {
1121 int status;
1122 TValue *o;
1123 lua_lock(L);
1124 api_checknelems(L, 1);
1125 o = s2v(L->top.p - 1);
1126 if (isLfunction(o))
1127 status = luaU_dump(L, getproto(o), writer, data, strip);
1128 else
1129 status = 1;
1130 lua_unlock(L);
1131 return status;
1132 }
1133
1134
lua_status(lua_State * L)1135 LUA_API int lua_status (lua_State *L) {
1136 return L->status;
1137 }
1138
1139
1140 /*
1141 ** Garbage-collection function
1142 */
lua_gc(lua_State * L,int what,...)1143 LUA_API int lua_gc (lua_State *L, int what, ...) {
1144 va_list argp;
1145 int res = 0;
1146 global_State *g = G(L);
1147 if (g->gcstp & GCSTPGC) /* internal stop? */
1148 return -1; /* all options are invalid when stopped */
1149 lua_lock(L);
1150 va_start(argp, what);
1151 switch (what) {
1152 case LUA_GCSTOP: {
1153 g->gcstp = GCSTPUSR; /* stopped by the user */
1154 break;
1155 }
1156 case LUA_GCRESTART: {
1157 luaE_setdebt(g, 0);
1158 g->gcstp = 0; /* (GCSTPGC must be already zero here) */
1159 break;
1160 }
1161 case LUA_GCCOLLECT: {
1162 luaC_fullgc(L, 0);
1163 break;
1164 }
1165 case LUA_GCCOUNT: {
1166 /* GC values are expressed in Kbytes: #bytes/2^10 */
1167 res = cast_int(gettotalbytes(g) >> 10);
1168 break;
1169 }
1170 case LUA_GCCOUNTB: {
1171 res = cast_int(gettotalbytes(g) & 0x3ff);
1172 break;
1173 }
1174 case LUA_GCSTEP: {
1175 int data = va_arg(argp, int);
1176 l_mem debt = 1; /* =1 to signal that it did an actual step */
1177 lu_byte oldstp = g->gcstp;
1178 g->gcstp = 0; /* allow GC to run (GCSTPGC must be zero here) */
1179 if (data == 0) {
1180 luaE_setdebt(g, 0); /* do a basic step */
1181 luaC_step(L);
1182 }
1183 else { /* add 'data' to total debt */
1184 debt = cast(l_mem, data) * 1024 + g->GCdebt;
1185 luaE_setdebt(g, debt);
1186 luaC_checkGC(L);
1187 }
1188 g->gcstp = oldstp; /* restore previous state */
1189 if (debt > 0 && g->gcstate == GCSpause) /* end of cycle? */
1190 res = 1; /* signal it */
1191 break;
1192 }
1193 case LUA_GCSETPAUSE: {
1194 int data = va_arg(argp, int);
1195 res = getgcparam(g->gcpause);
1196 setgcparam(g->gcpause, data);
1197 break;
1198 }
1199 case LUA_GCSETSTEPMUL: {
1200 int data = va_arg(argp, int);
1201 res = getgcparam(g->gcstepmul);
1202 setgcparam(g->gcstepmul, data);
1203 break;
1204 }
1205 case LUA_GCISRUNNING: {
1206 res = gcrunning(g);
1207 break;
1208 }
1209 case LUA_GCGEN: {
1210 int minormul = va_arg(argp, int);
1211 int majormul = va_arg(argp, int);
1212 res = isdecGCmodegen(g) ? LUA_GCGEN : LUA_GCINC;
1213 if (minormul != 0)
1214 g->genminormul = minormul;
1215 if (majormul != 0)
1216 setgcparam(g->genmajormul, majormul);
1217 luaC_changemode(L, KGC_GEN);
1218 break;
1219 }
1220 case LUA_GCINC: {
1221 int pause = va_arg(argp, int);
1222 int stepmul = va_arg(argp, int);
1223 int stepsize = va_arg(argp, int);
1224 res = isdecGCmodegen(g) ? LUA_GCGEN : LUA_GCINC;
1225 if (pause != 0)
1226 setgcparam(g->gcpause, pause);
1227 if (stepmul != 0)
1228 setgcparam(g->gcstepmul, stepmul);
1229 if (stepsize != 0)
1230 g->gcstepsize = stepsize;
1231 luaC_changemode(L, KGC_INC);
1232 break;
1233 }
1234 default: res = -1; /* invalid option */
1235 }
1236 va_end(argp);
1237 lua_unlock(L);
1238 return res;
1239 }
1240
1241
1242
1243 /*
1244 ** miscellaneous functions
1245 */
1246
1247
lua_error(lua_State * L)1248 LUA_API int lua_error (lua_State *L) {
1249 TValue *errobj;
1250 lua_lock(L);
1251 errobj = s2v(L->top.p - 1);
1252 api_checknelems(L, 1);
1253 /* error object is the memory error message? */
1254 if (ttisshrstring(errobj) && eqshrstr(tsvalue(errobj), G(L)->memerrmsg))
1255 luaM_error(L); /* raise a memory error */
1256 else
1257 luaG_errormsg(L); /* raise a regular error */
1258 /* code unreachable; will unlock when control actually leaves the kernel */
1259 return 0; /* to avoid warnings */
1260 }
1261
1262
lua_next(lua_State * L,int idx)1263 LUA_API int lua_next (lua_State *L, int idx) {
1264 Table *t;
1265 int more;
1266 lua_lock(L);
1267 api_checknelems(L, 1);
1268 t = gettable(L, idx);
1269 more = luaH_next(L, t, L->top.p - 1);
1270 if (more) {
1271 api_incr_top(L);
1272 }
1273 else /* no more elements */
1274 L->top.p -= 1; /* remove key */
1275 lua_unlock(L);
1276 return more;
1277 }
1278
1279
lua_toclose(lua_State * L,int idx)1280 LUA_API void lua_toclose (lua_State *L, int idx) {
1281 int nresults;
1282 StkId o;
1283 lua_lock(L);
1284 o = index2stack(L, idx);
1285 nresults = L->ci->nresults;
1286 api_check(L, L->tbclist.p < o, "given index below or equal a marked one");
1287 luaF_newtbcupval(L, o); /* create new to-be-closed upvalue */
1288 if (!hastocloseCfunc(nresults)) /* function not marked yet? */
1289 L->ci->nresults = codeNresults(nresults); /* mark it */
1290 lua_assert(hastocloseCfunc(L->ci->nresults));
1291 lua_unlock(L);
1292 }
1293
1294
lua_concat(lua_State * L,int n)1295 LUA_API void lua_concat (lua_State *L, int n) {
1296 lua_lock(L);
1297 api_checknelems(L, n);
1298 if (n > 0)
1299 luaV_concat(L, n);
1300 else { /* nothing to concatenate */
1301 setsvalue2s(L, L->top.p, luaS_newlstr(L, "", 0)); /* push empty string */
1302 api_incr_top(L);
1303 }
1304 luaC_checkGC(L);
1305 lua_unlock(L);
1306 }
1307
1308
lua_len(lua_State * L,int idx)1309 LUA_API void lua_len (lua_State *L, int idx) {
1310 TValue *t;
1311 lua_lock(L);
1312 t = index2value(L, idx);
1313 luaV_objlen(L, L->top.p, t);
1314 api_incr_top(L);
1315 lua_unlock(L);
1316 }
1317
1318
lua_getallocf(lua_State * L,void ** ud)1319 LUA_API lua_Alloc lua_getallocf (lua_State *L, void **ud) {
1320 lua_Alloc f;
1321 lua_lock(L);
1322 if (ud) *ud = G(L)->ud;
1323 f = G(L)->frealloc;
1324 lua_unlock(L);
1325 return f;
1326 }
1327
1328
lua_setallocf(lua_State * L,lua_Alloc f,void * ud)1329 LUA_API void lua_setallocf (lua_State *L, lua_Alloc f, void *ud) {
1330 lua_lock(L);
1331 G(L)->ud = ud;
1332 G(L)->frealloc = f;
1333 lua_unlock(L);
1334 }
1335
1336
lua_setwarnf(lua_State * L,lua_WarnFunction f,void * ud)1337 void lua_setwarnf (lua_State *L, lua_WarnFunction f, void *ud) {
1338 lua_lock(L);
1339 G(L)->ud_warn = ud;
1340 G(L)->warnf = f;
1341 lua_unlock(L);
1342 }
1343
1344
lua_warning(lua_State * L,const char * msg,int tocont)1345 void lua_warning (lua_State *L, const char *msg, int tocont) {
1346 lua_lock(L);
1347 luaE_warning(L, msg, tocont);
1348 lua_unlock(L);
1349 }
1350
1351
1352
lua_newuserdatauv(lua_State * L,size_t size,int nuvalue)1353 LUA_API void *lua_newuserdatauv (lua_State *L, size_t size, int nuvalue) {
1354 Udata *u;
1355 lua_lock(L);
1356 api_check(L, 0 <= nuvalue && nuvalue < USHRT_MAX, "invalid value");
1357 u = luaS_newudata(L, size, nuvalue);
1358 setuvalue(L, s2v(L->top.p), u);
1359 api_incr_top(L);
1360 luaC_checkGC(L);
1361 lua_unlock(L);
1362 return getudatamem(u);
1363 }
1364
1365
1366
aux_upvalue(TValue * fi,int n,TValue ** val,GCObject ** owner)1367 static const char *aux_upvalue (TValue *fi, int n, TValue **val,
1368 GCObject **owner) {
1369 switch (ttypetag(fi)) {
1370 case LUA_VCCL: { /* C closure */
1371 CClosure *f = clCvalue(fi);
1372 if (!(cast_uint(n) - 1u < cast_uint(f->nupvalues)))
1373 return NULL; /* 'n' not in [1, f->nupvalues] */
1374 *val = &f->upvalue[n-1];
1375 if (owner) *owner = obj2gco(f);
1376 return "";
1377 }
1378 case LUA_VLCL: { /* Lua closure */
1379 LClosure *f = clLvalue(fi);
1380 TString *name;
1381 Proto *p = f->p;
1382 if (!(cast_uint(n) - 1u < cast_uint(p->sizeupvalues)))
1383 return NULL; /* 'n' not in [1, p->sizeupvalues] */
1384 *val = f->upvals[n-1]->v.p;
1385 if (owner) *owner = obj2gco(f->upvals[n - 1]);
1386 name = p->upvalues[n-1].name;
1387 return (name == NULL) ? "(no name)" : getstr(name);
1388 }
1389 default: return NULL; /* not a closure */
1390 }
1391 }
1392
1393
lua_getupvalue(lua_State * L,int funcindex,int n)1394 LUA_API const char *lua_getupvalue (lua_State *L, int funcindex, int n) {
1395 const char *name;
1396 TValue *val = NULL; /* to avoid warnings */
1397 lua_lock(L);
1398 name = aux_upvalue(index2value(L, funcindex), n, &val, NULL);
1399 if (name) {
1400 setobj2s(L, L->top.p, val);
1401 api_incr_top(L);
1402 }
1403 lua_unlock(L);
1404 return name;
1405 }
1406
1407
lua_setupvalue(lua_State * L,int funcindex,int n)1408 LUA_API const char *lua_setupvalue (lua_State *L, int funcindex, int n) {
1409 const char *name;
1410 TValue *val = NULL; /* to avoid warnings */
1411 GCObject *owner = NULL; /* to avoid warnings */
1412 TValue *fi;
1413 lua_lock(L);
1414 fi = index2value(L, funcindex);
1415 api_checknelems(L, 1);
1416 name = aux_upvalue(fi, n, &val, &owner);
1417 if (name) {
1418 L->top.p--;
1419 setobj(L, val, s2v(L->top.p));
1420 luaC_barrier(L, owner, val);
1421 }
1422 lua_unlock(L);
1423 return name;
1424 }
1425
1426
getupvalref(lua_State * L,int fidx,int n,LClosure ** pf)1427 static UpVal **getupvalref (lua_State *L, int fidx, int n, LClosure **pf) {
1428 static const UpVal *const nullup = NULL;
1429 LClosure *f;
1430 TValue *fi = index2value(L, fidx);
1431 api_check(L, ttisLclosure(fi), "Lua function expected");
1432 f = clLvalue(fi);
1433 if (pf) *pf = f;
1434 if (1 <= n && n <= f->p->sizeupvalues)
1435 return &f->upvals[n - 1]; /* get its upvalue pointer */
1436 else
1437 return (UpVal**)&nullup;
1438 }
1439
1440
lua_upvalueid(lua_State * L,int fidx,int n)1441 LUA_API void *lua_upvalueid (lua_State *L, int fidx, int n) {
1442 TValue *fi = index2value(L, fidx);
1443 switch (ttypetag(fi)) {
1444 case LUA_VLCL: { /* lua closure */
1445 return *getupvalref(L, fidx, n, NULL);
1446 }
1447 case LUA_VCCL: { /* C closure */
1448 CClosure *f = clCvalue(fi);
1449 if (1 <= n && n <= f->nupvalues)
1450 return &f->upvalue[n - 1];
1451 /* else */
1452 } /* FALLTHROUGH */
1453 case LUA_VLCF:
1454 return NULL; /* light C functions have no upvalues */
1455 default: {
1456 api_check(L, 0, "function expected");
1457 return NULL;
1458 }
1459 }
1460 }
1461
1462
lua_upvaluejoin(lua_State * L,int fidx1,int n1,int fidx2,int n2)1463 LUA_API void lua_upvaluejoin (lua_State *L, int fidx1, int n1,
1464 int fidx2, int n2) {
1465 LClosure *f1;
1466 UpVal **up1 = getupvalref(L, fidx1, n1, &f1);
1467 UpVal **up2 = getupvalref(L, fidx2, n2, NULL);
1468 api_check(L, *up1 != NULL && *up2 != NULL, "invalid upvalue index");
1469 *up1 = *up2;
1470 luaC_objbarrier(L, f1, *up1);
1471 }
1472
1473
1474