mirror of https://github.com/rusefi/lua.git
First "complete" implementation of to-be-closed variables
Still missing: - handling of memory errors when creating upvalue (must run closing method all the same) - interaction with coroutines
This commit is contained in:
parent
4cd1f4aac0
commit
bd96330d03
20
ldo.c
20
ldo.c
|
@ -88,7 +88,7 @@ struct lua_longjmp {
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
static void seterrorobj (lua_State *L, int errcode, StkId oldtop) {
|
void luaD_seterrorobj (lua_State *L, int errcode, StkId oldtop) {
|
||||||
switch (errcode) {
|
switch (errcode) {
|
||||||
case LUA_ERRMEM: { /* memory error? */
|
case LUA_ERRMEM: { /* memory error? */
|
||||||
setsvalue2s(L, oldtop, G(L)->memerrmsg); /* reuse preregistered msg. */
|
setsvalue2s(L, oldtop, G(L)->memerrmsg); /* reuse preregistered msg. */
|
||||||
|
@ -121,7 +121,7 @@ l_noret luaD_throw (lua_State *L, int errcode) {
|
||||||
}
|
}
|
||||||
else { /* no handler at all; abort */
|
else { /* no handler at all; abort */
|
||||||
if (g->panic) { /* panic function? */
|
if (g->panic) { /* panic function? */
|
||||||
seterrorobj(L, errcode, L->top); /* assume EXTRA_STACK */
|
luaD_seterrorobj(L, errcode, L->top); /* assume EXTRA_STACK */
|
||||||
if (L->ci->top < L->top)
|
if (L->ci->top < L->top)
|
||||||
L->ci->top = L->top; /* pushing msg. can break this invariant */
|
L->ci->top = L->top; /* pushing msg. can break this invariant */
|
||||||
lua_unlock(L);
|
lua_unlock(L);
|
||||||
|
@ -584,8 +584,8 @@ static int recover (lua_State *L, int status) {
|
||||||
if (ci == NULL) return 0; /* no recovery point */
|
if (ci == NULL) return 0; /* no recovery point */
|
||||||
/* "finish" luaD_pcall */
|
/* "finish" luaD_pcall */
|
||||||
oldtop = restorestack(L, ci->u2.funcidx);
|
oldtop = restorestack(L, ci->u2.funcidx);
|
||||||
luaF_close(L, oldtop);
|
luaF_close(L, oldtop, status);
|
||||||
seterrorobj(L, status, oldtop);
|
luaD_seterrorobj(L, status, oldtop);
|
||||||
L->ci = ci;
|
L->ci = ci;
|
||||||
L->allowhook = getoah(ci->callstatus); /* restore original 'allowhook' */
|
L->allowhook = getoah(ci->callstatus); /* restore original 'allowhook' */
|
||||||
L->nny = 0; /* should be zero to be yieldable */
|
L->nny = 0; /* should be zero to be yieldable */
|
||||||
|
@ -678,7 +678,7 @@ LUA_API int lua_resume (lua_State *L, lua_State *from, int nargs,
|
||||||
}
|
}
|
||||||
if (unlikely(errorstatus(status))) { /* unrecoverable error? */
|
if (unlikely(errorstatus(status))) { /* unrecoverable error? */
|
||||||
L->status = cast_byte(status); /* mark thread as 'dead' */
|
L->status = cast_byte(status); /* mark thread as 'dead' */
|
||||||
seterrorobj(L, status, L->top); /* push error message */
|
luaD_seterrorobj(L, status, L->top); /* push error message */
|
||||||
L->ci->top = L->top;
|
L->ci->top = L->top;
|
||||||
}
|
}
|
||||||
else lua_assert(status == L->status); /* normal end or yield */
|
else lua_assert(status == L->status); /* normal end or yield */
|
||||||
|
@ -726,6 +726,11 @@ LUA_API int lua_yieldk (lua_State *L, int nresults, lua_KContext ctx,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
** Call the C function 'func' in protected mode, restoring basic
|
||||||
|
** thread information ('allowhook', 'nny', etc.) and in particular
|
||||||
|
** its stack level in case of errors.
|
||||||
|
*/
|
||||||
int luaD_pcall (lua_State *L, Pfunc func, void *u,
|
int luaD_pcall (lua_State *L, Pfunc func, void *u,
|
||||||
ptrdiff_t old_top, ptrdiff_t ef) {
|
ptrdiff_t old_top, ptrdiff_t ef) {
|
||||||
int status;
|
int status;
|
||||||
|
@ -737,11 +742,12 @@ int luaD_pcall (lua_State *L, Pfunc func, void *u,
|
||||||
status = luaD_rawrunprotected(L, func, u);
|
status = luaD_rawrunprotected(L, func, u);
|
||||||
if (unlikely(status != LUA_OK)) { /* an error occurred? */
|
if (unlikely(status != LUA_OK)) { /* an error occurred? */
|
||||||
StkId oldtop = restorestack(L, old_top);
|
StkId oldtop = restorestack(L, old_top);
|
||||||
luaF_close(L, oldtop); /* close possible pending closures */
|
|
||||||
seterrorobj(L, status, oldtop);
|
|
||||||
L->ci = old_ci;
|
L->ci = old_ci;
|
||||||
L->allowhook = old_allowhooks;
|
L->allowhook = old_allowhooks;
|
||||||
L->nny = old_nny;
|
L->nny = old_nny;
|
||||||
|
status = luaF_close(L, oldtop, status);
|
||||||
|
oldtop = restorestack(L, old_top); /* previous call may change stack */
|
||||||
|
luaD_seterrorobj(L, status, oldtop);
|
||||||
luaD_shrinkstack(L);
|
luaD_shrinkstack(L);
|
||||||
}
|
}
|
||||||
L->errfunc = old_errfunc;
|
L->errfunc = old_errfunc;
|
||||||
|
|
1
ldo.h
1
ldo.h
|
@ -50,6 +50,7 @@
|
||||||
/* type of protected functions, to be ran by 'runprotected' */
|
/* type of protected functions, to be ran by 'runprotected' */
|
||||||
typedef void (*Pfunc) (lua_State *L, void *ud);
|
typedef void (*Pfunc) (lua_State *L, void *ud);
|
||||||
|
|
||||||
|
LUAI_FUNC void luaD_seterrorobj (lua_State *L, int errcode, StkId oldtop);
|
||||||
LUAI_FUNC int luaD_protectedparser (lua_State *L, ZIO *z, const char *name,
|
LUAI_FUNC int luaD_protectedparser (lua_State *L, ZIO *z, const char *name,
|
||||||
const char *mode);
|
const char *mode);
|
||||||
LUAI_FUNC void luaD_hook (lua_State *L, int event, int line,
|
LUAI_FUNC void luaD_hook (lua_State *L, int event, int line,
|
||||||
|
|
47
lfunc.c
47
lfunc.c
|
@ -14,6 +14,7 @@
|
||||||
|
|
||||||
#include "lua.h"
|
#include "lua.h"
|
||||||
|
|
||||||
|
#include "ldo.h"
|
||||||
#include "lfunc.h"
|
#include "lfunc.h"
|
||||||
#include "lgc.h"
|
#include "lgc.h"
|
||||||
#include "lmem.h"
|
#include "lmem.h"
|
||||||
|
@ -83,6 +84,40 @@ UpVal *luaF_findupval (lua_State *L, StkId level) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void callclose (lua_State *L, void *ud) {
|
||||||
|
luaD_callnoyield(L, cast(StkId, ud), 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int closeupval (lua_State *L, UpVal *uv, StkId level, int status) {
|
||||||
|
StkId func = level + 1; /* save slot for old error message */
|
||||||
|
if (status != LUA_OK) /* was there an error? */
|
||||||
|
luaD_seterrorobj(L, status, level); /* save error message */
|
||||||
|
else
|
||||||
|
setnilvalue(s2v(level));
|
||||||
|
if (ttisfunction(uv->v)) { /* object to-be-closed is a function? */
|
||||||
|
setobj2s(L, func, uv->v); /* will call it */
|
||||||
|
setobjs2s(L, func + 1, level); /* error msg. as argument */
|
||||||
|
}
|
||||||
|
else { /* try '__close' metamethod */
|
||||||
|
const TValue *tm = luaT_gettmbyobj(L, uv->v, TM_CLOSE);
|
||||||
|
if (ttisnil(tm))
|
||||||
|
return status; /* no metamethod */
|
||||||
|
setobj2s(L, func, tm); /* will call metamethod */
|
||||||
|
setobj2s(L, func + 1, uv->v); /* with 'self' as argument */
|
||||||
|
}
|
||||||
|
L->top = func + 2; /* add function and argument */
|
||||||
|
if (status == LUA_OK) /* not in "error mode"? */
|
||||||
|
callclose(L, func); /* call closing method */
|
||||||
|
else { /* already inside error handler; cannot raise another error */
|
||||||
|
int newstatus = luaD_pcall(L, callclose, func, savestack(L, level), 0);
|
||||||
|
if (newstatus != LUA_OK) /* error when closing? */
|
||||||
|
status = newstatus; /* this will be the new error */
|
||||||
|
}
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
void luaF_unlinkupval (UpVal *uv) {
|
void luaF_unlinkupval (UpVal *uv) {
|
||||||
lua_assert(upisopen(uv));
|
lua_assert(upisopen(uv));
|
||||||
*uv->u.open.previous = uv->u.open.next;
|
*uv->u.open.previous = uv->u.open.next;
|
||||||
|
@ -91,10 +126,10 @@ void luaF_unlinkupval (UpVal *uv) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void luaF_close (lua_State *L, StkId level) {
|
int luaF_close (lua_State *L, StkId level, int status) {
|
||||||
UpVal *uv;
|
UpVal *uv;
|
||||||
while (L->openupval != NULL &&
|
while ((uv = L->openupval) != NULL && uplevel(uv) >= level) {
|
||||||
(uv = L->openupval, uplevel(uv) >= level)) {
|
StkId upl = uplevel(uv);
|
||||||
TValue *slot = &uv->u.value; /* new position for value */
|
TValue *slot = &uv->u.value; /* new position for value */
|
||||||
luaF_unlinkupval(uv);
|
luaF_unlinkupval(uv);
|
||||||
setobj(L, slot, uv->v); /* move value to upvalue slot */
|
setobj(L, slot, uv->v); /* move value to upvalue slot */
|
||||||
|
@ -102,7 +137,13 @@ void luaF_close (lua_State *L, StkId level) {
|
||||||
if (!iswhite(uv))
|
if (!iswhite(uv))
|
||||||
gray2black(uv); /* closed upvalues cannot be gray */
|
gray2black(uv); /* closed upvalues cannot be gray */
|
||||||
luaC_barrier(L, uv, slot);
|
luaC_barrier(L, uv, slot);
|
||||||
|
if (status >= 0 && uv->tt == LUA_TUPVALTBC) { /* must be closed? */
|
||||||
|
ptrdiff_t levelrel = savestack(L, level);
|
||||||
|
status = closeupval(L, uv, upl, status); /* may reallocate the stack */
|
||||||
|
level = restorestack(L, levelrel);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
2
lfunc.h
2
lfunc.h
|
@ -47,7 +47,7 @@ LUAI_FUNC CClosure *luaF_newCclosure (lua_State *L, int nelems);
|
||||||
LUAI_FUNC LClosure *luaF_newLclosure (lua_State *L, int nelems);
|
LUAI_FUNC LClosure *luaF_newLclosure (lua_State *L, int nelems);
|
||||||
LUAI_FUNC void luaF_initupvals (lua_State *L, LClosure *cl);
|
LUAI_FUNC void luaF_initupvals (lua_State *L, LClosure *cl);
|
||||||
LUAI_FUNC UpVal *luaF_findupval (lua_State *L, StkId level);
|
LUAI_FUNC UpVal *luaF_findupval (lua_State *L, StkId level);
|
||||||
LUAI_FUNC void luaF_close (lua_State *L, StkId level);
|
LUAI_FUNC int luaF_close (lua_State *L, StkId level, int status);
|
||||||
LUAI_FUNC void luaF_unlinkupval (UpVal *uv);
|
LUAI_FUNC void luaF_unlinkupval (UpVal *uv);
|
||||||
LUAI_FUNC void luaF_freeproto (lua_State *L, Proto *f);
|
LUAI_FUNC void luaF_freeproto (lua_State *L, Proto *f);
|
||||||
LUAI_FUNC const char *luaF_getlocalname (const Proto *func, int local_number,
|
LUAI_FUNC const char *luaF_getlocalname (const Proto *func, int local_number,
|
||||||
|
|
5
lgc.c
5
lgc.c
|
@ -609,6 +609,7 @@ static int traverseLclosure (global_State *g, LClosure *cl) {
|
||||||
** That ensures that the entire stack have valid (non-dead) objects.
|
** That ensures that the entire stack have valid (non-dead) objects.
|
||||||
*/
|
*/
|
||||||
static int traversethread (global_State *g, lua_State *th) {
|
static int traversethread (global_State *g, lua_State *th) {
|
||||||
|
UpVal *uv;
|
||||||
StkId o = th->stack;
|
StkId o = th->stack;
|
||||||
if (o == NULL)
|
if (o == NULL)
|
||||||
return 1; /* stack not completely built yet */
|
return 1; /* stack not completely built yet */
|
||||||
|
@ -616,6 +617,10 @@ static int traversethread (global_State *g, lua_State *th) {
|
||||||
th->openupval == NULL || isintwups(th));
|
th->openupval == NULL || isintwups(th));
|
||||||
for (; o < th->top; o++) /* mark live elements in the stack */
|
for (; o < th->top; o++) /* mark live elements in the stack */
|
||||||
markvalue(g, s2v(o));
|
markvalue(g, s2v(o));
|
||||||
|
for (uv = th->openupval; uv != NULL; uv = uv->u.open.next) {
|
||||||
|
if (uv->tt == LUA_TUPVALTBC) /* to be closed? */
|
||||||
|
markobject(g, uv); /* cannot be collected */
|
||||||
|
}
|
||||||
if (g->gcstate == GCSatomic) { /* final traversal? */
|
if (g->gcstate == GCSatomic) { /* final traversal? */
|
||||||
StkId lim = th->stack + th->stacksize; /* real end of stack */
|
StkId lim = th->stack + th->stacksize; /* real end of stack */
|
||||||
for (; o < lim; o++) /* clear not-marked stack slice */
|
for (; o < lim; o++) /* clear not-marked stack slice */
|
||||||
|
|
|
@ -1536,9 +1536,9 @@ static void scopedlocalstat (LexState *ls) {
|
||||||
FuncState *fs = ls->fs;
|
FuncState *fs = ls->fs;
|
||||||
new_localvar(ls, str_checkname(ls));
|
new_localvar(ls, str_checkname(ls));
|
||||||
checknext(ls, '=');
|
checknext(ls, '=');
|
||||||
|
exp1(ls, 0);
|
||||||
luaK_codeABC(fs, OP_TBC, fs->nactvar, 0, 0);
|
luaK_codeABC(fs, OP_TBC, fs->nactvar, 0, 0);
|
||||||
markupval(fs, fs->nactvar);
|
markupval(fs, fs->nactvar);
|
||||||
exp1(ls, 0);
|
|
||||||
adjustlocalvars(ls, 1);
|
adjustlocalvars(ls, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
4
lstate.c
4
lstate.c
|
@ -258,7 +258,7 @@ static void preinit_thread (lua_State *L, global_State *g) {
|
||||||
|
|
||||||
static void close_state (lua_State *L) {
|
static void close_state (lua_State *L) {
|
||||||
global_State *g = G(L);
|
global_State *g = G(L);
|
||||||
luaF_close(L, L->stack); /* close all upvalues for this thread */
|
luaF_close(L, L->stack, -1); /* close all upvalues for this thread */
|
||||||
luaC_freeallobjects(L); /* collect all objects */
|
luaC_freeallobjects(L); /* collect all objects */
|
||||||
if (ttisnil(&g->nilvalue)) /* closing a fully built state? */
|
if (ttisnil(&g->nilvalue)) /* closing a fully built state? */
|
||||||
luai_userstateclose(L);
|
luai_userstateclose(L);
|
||||||
|
@ -301,7 +301,7 @@ LUA_API lua_State *lua_newthread (lua_State *L) {
|
||||||
|
|
||||||
void luaE_freethread (lua_State *L, lua_State *L1) {
|
void luaE_freethread (lua_State *L, lua_State *L1) {
|
||||||
LX *l = fromstate(L1);
|
LX *l = fromstate(L1);
|
||||||
luaF_close(L1, L1->stack); /* close all upvalues for this thread */
|
luaF_close(L1, L1->stack, -1); /* close all upvalues for this thread */
|
||||||
lua_assert(L1->openupval == NULL);
|
lua_assert(L1->openupval == NULL);
|
||||||
luai_userstatefree(L, L1);
|
luai_userstatefree(L, L1);
|
||||||
freestack(L1);
|
freestack(L1);
|
||||||
|
|
2
ltests.c
2
ltests.c
|
@ -1208,7 +1208,7 @@ static int getindex_aux (lua_State *L, lua_State *L1, const char **pc) {
|
||||||
|
|
||||||
static void pushcode (lua_State *L, int code) {
|
static void pushcode (lua_State *L, int code) {
|
||||||
static const char *const codes[] = {"OK", "YIELD", "ERRRUN",
|
static const char *const codes[] = {"OK", "YIELD", "ERRRUN",
|
||||||
"ERRSYNTAX", "ERRMEM", "ERRGCMM", "ERRERR"};
|
"ERRSYNTAX", MEMERRMSG, "ERRGCMM", "ERRERR"};
|
||||||
lua_pushstring(L, codes[code]);
|
lua_pushstring(L, codes[code]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
2
ltm.c
2
ltm.c
|
@ -43,7 +43,7 @@ void luaT_init (lua_State *L) {
|
||||||
"__div", "__idiv",
|
"__div", "__idiv",
|
||||||
"__band", "__bor", "__bxor", "__shl", "__shr",
|
"__band", "__bor", "__bxor", "__shl", "__shr",
|
||||||
"__unm", "__bnot", "__lt", "__le",
|
"__unm", "__bnot", "__lt", "__le",
|
||||||
"__concat", "__call"
|
"__concat", "__call", "__close"
|
||||||
};
|
};
|
||||||
int i;
|
int i;
|
||||||
for (i=0; i<TM_N; i++) {
|
for (i=0; i<TM_N; i++) {
|
||||||
|
|
1
ltm.h
1
ltm.h
|
@ -40,6 +40,7 @@ typedef enum {
|
||||||
TM_LE,
|
TM_LE,
|
||||||
TM_CONCAT,
|
TM_CONCAT,
|
||||||
TM_CALL,
|
TM_CALL,
|
||||||
|
TM_CLOSE,
|
||||||
TM_N /* number of elements in the enum */
|
TM_N /* number of elements in the enum */
|
||||||
} TMS;
|
} TMS;
|
||||||
|
|
||||||
|
|
7
lvm.c
7
lvm.c
|
@ -1452,13 +1452,12 @@ void luaV_execute (lua_State *L, CallInfo *ci) {
|
||||||
vmbreak;
|
vmbreak;
|
||||||
}
|
}
|
||||||
vmcase(OP_CLOSE) {
|
vmcase(OP_CLOSE) {
|
||||||
luaF_close(L, ra);
|
luaF_close(L, ra, LUA_OK);
|
||||||
vmbreak;
|
vmbreak;
|
||||||
}
|
}
|
||||||
vmcase(OP_TBC) {
|
vmcase(OP_TBC) {
|
||||||
UpVal *up = luaF_findupval(L, ra); /* create new upvalue */
|
UpVal *up = luaF_findupval(L, ra); /* create new upvalue */
|
||||||
up->tt = LUA_TUPVALTBC; /* mark it to be closed */
|
up->tt = LUA_TUPVALTBC; /* mark it to be closed */
|
||||||
setnilvalue(s2v(ra)); /* intialize it with nil */
|
|
||||||
vmbreak;
|
vmbreak;
|
||||||
}
|
}
|
||||||
vmcase(OP_JMP) {
|
vmcase(OP_JMP) {
|
||||||
|
@ -1591,7 +1590,7 @@ void luaV_execute (lua_State *L, CallInfo *ci) {
|
||||||
int nparams1 = GETARG_C(i);
|
int nparams1 = GETARG_C(i);
|
||||||
if (nparams1) /* vararg function? */
|
if (nparams1) /* vararg function? */
|
||||||
delta = ci->u.l.nextraargs + nparams1;
|
delta = ci->u.l.nextraargs + nparams1;
|
||||||
luaF_close(L, base); /* close upvalues from current call */
|
luaF_close(L, base, LUA_OK); /* close upvalues from current call */
|
||||||
}
|
}
|
||||||
if (!ttisfunction(s2v(ra))) { /* not a function? */
|
if (!ttisfunction(s2v(ra))) { /* not a function? */
|
||||||
luaD_tryfuncTM(L, ra); /* try '__call' metamethod */
|
luaD_tryfuncTM(L, ra); /* try '__call' metamethod */
|
||||||
|
@ -1625,7 +1624,7 @@ void luaV_execute (lua_State *L, CallInfo *ci) {
|
||||||
int nparams1 = GETARG_C(i);
|
int nparams1 = GETARG_C(i);
|
||||||
if (nparams1) /* vararg function? */
|
if (nparams1) /* vararg function? */
|
||||||
ci->func -= ci->u.l.nextraargs + nparams1;
|
ci->func -= ci->u.l.nextraargs + nparams1;
|
||||||
luaF_close(L, base); /* there may be open upvalues */
|
luaF_close(L, base, LUA_OK); /* there may be open upvalues */
|
||||||
}
|
}
|
||||||
halfProtect(luaD_poscall(L, ci, n));
|
halfProtect(luaD_poscall(L, ci, n));
|
||||||
return;
|
return;
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
-- $Id: testes/api.lua $
|
-- $Id: testes/api.lua 2018-07-25 15:31:04 -0300 $
|
||||||
-- See Copyright Notice in file all.lua
|
-- See Copyright Notice in file all.lua
|
||||||
|
|
||||||
if T==nil then
|
if T==nil then
|
||||||
|
@ -1027,6 +1027,18 @@ testamem("coroutine creation", function()
|
||||||
end)
|
end)
|
||||||
|
|
||||||
|
|
||||||
|
-- testing to-be-closed variables
|
||||||
|
testamem("to-be-closed variables", function()
|
||||||
|
local flag
|
||||||
|
do
|
||||||
|
local scoped x = function () flag = true end
|
||||||
|
flag = false
|
||||||
|
local x = {}
|
||||||
|
end
|
||||||
|
return flag
|
||||||
|
end)
|
||||||
|
|
||||||
|
|
||||||
-- testing threads
|
-- testing threads
|
||||||
|
|
||||||
-- get main thread from registry (at index LUA_RIDX_MAINTHREAD == 1)
|
-- get main thread from registry (at index LUA_RIDX_MAINTHREAD == 1)
|
||||||
|
|
|
@ -173,15 +173,69 @@ end
|
||||||
assert(x==20)
|
assert(x==20)
|
||||||
|
|
||||||
|
|
||||||
-- tests for to-be-closed variables
|
print"testing to-be-closed variables"
|
||||||
|
|
||||||
do
|
do
|
||||||
local scoped x = 3
|
local a = {}
|
||||||
local a
|
do
|
||||||
local scoped y = 5
|
local scoped x = setmetatable({"x"}, {__close = function (self)
|
||||||
assert(x == 3 and y == 5)
|
a[#a + 1] = self[1] end})
|
||||||
|
local scoped y = function () a[#a + 1] = "y" end
|
||||||
|
a[#a + 1] = "in"
|
||||||
|
end
|
||||||
|
a[#a + 1] = "out"
|
||||||
|
assert(a[1] == "in" and a[2] == "y" and a[3] == "x" and a[4] == "out")
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
|
do -- errors in __close
|
||||||
|
local log = {}
|
||||||
|
local function foo (err)
|
||||||
|
local scoped x = function (msg) log[#log + 1] = msg; error(1) end
|
||||||
|
local scoped x1 = function (msg) log[#log + 1] = msg; end
|
||||||
|
local scoped gc = function () collectgarbage() end
|
||||||
|
local scoped y = function (msg) log[#log + 1] = msg; error(2) end
|
||||||
|
local scoped z = function (msg) log[#log + 1] = msg or 10; error(3) end
|
||||||
|
if err then error(4) end
|
||||||
|
end
|
||||||
|
local stat, msg = pcall(foo, false)
|
||||||
|
assert(msg == 1)
|
||||||
|
assert(log[1] == 10 and log[2] == 3 and log[3] == 2 and log[4] == 2
|
||||||
|
and #log == 4)
|
||||||
|
|
||||||
|
log = {}
|
||||||
|
local stat, msg = pcall(foo, true)
|
||||||
|
assert(msg == 1)
|
||||||
|
assert(log[1] == 4 and log[2] == 3 and log[3] == 2 and log[4] == 2
|
||||||
|
and #log == 4)
|
||||||
|
end
|
||||||
|
|
||||||
|
do
|
||||||
|
-- memory error inside closing function
|
||||||
|
local function foo ()
|
||||||
|
local scoped y = function () io.write(2); T.alloccount() end
|
||||||
|
local scoped x = setmetatable({}, {__close = function ()
|
||||||
|
T.alloccount(0); local x = {} -- force a memory error
|
||||||
|
end})
|
||||||
|
io.write("1\n")
|
||||||
|
error("a") -- common error inside the function's body
|
||||||
|
end
|
||||||
|
|
||||||
|
local _, msg = pcall(foo)
|
||||||
|
T.alloccount()
|
||||||
|
assert(msg == "not enough memory")
|
||||||
|
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
-- a suspended coroutine should not close its variables when collected
|
||||||
|
local co = coroutine.wrap(function()
|
||||||
|
local scoped x = function () os.exit(1) end -- should not run
|
||||||
|
coroutine.yield()
|
||||||
|
end)
|
||||||
|
co()
|
||||||
|
co = nil
|
||||||
|
|
||||||
print('OK')
|
print('OK')
|
||||||
|
|
||||||
return 5,f
|
return 5,f
|
||||||
|
|
Loading…
Reference in New Issue