refactoring of common parts of 'pcall' and 'xpcall' + avoids

stack errors on these functions (return error code instead)
This commit is contained in:
Roberto Ierusalimschy 2011-11-09 17:28:27 -02:00
parent eef22fa4ce
commit 888d39ea75
1 changed files with 21 additions and 16 deletions

View File

@ -1,5 +1,5 @@
/* /*
** $Id: lbaselib.c,v 1.265 2011/07/27 12:14:06 roberto Exp roberto $ ** $Id: lbaselib.c,v 1.266 2011/09/30 12:43:54 roberto Exp roberto $
** Basic library ** Basic library
** See Copyright Notice in lua.h ** See Copyright Notice in lua.h
*/ */
@ -391,27 +391,35 @@ static int luaB_select (lua_State *L) {
} }
static int pcallcont (lua_State *L) { static int finishpcall (lua_State *L, int status, int isx) {
int errfunc = 0; /* =0 to avoid warnings */ if (!lua_checkstack(L, 1)) { /* no space for extra boolean? */
int status = lua_getctx(L, &errfunc); lua_settop(L, 0); /* create space for return values */
lua_assert(status != LUA_OK); lua_pushboolean(L, 0);
lua_pushboolean(L, (status == LUA_YIELD)); /* first result (status) */ lua_pushstring(L, "stack overflow");
if (errfunc) /* came from xpcall? */ return 2; /* return false, msg */
}
lua_pushboolean(L, status); /* first result (status) */
if (isx) /* came from xpcall? */
lua_replace(L, 1); /* put first result in place of error function */ lua_replace(L, 1); /* put first result in place of error function */
else /* came from pcall */ else /* came from pcall */
lua_insert(L, 1); /* open space for first result */ lua_insert(L, 1); /* insert first result before the others */
return lua_gettop(L); return lua_gettop(L);
} }
static int pcallcont (lua_State *L) {
int isx = 0; /* =0 to avoid warnings */
int status = lua_getctx(L, &isx);
lua_assert(status != LUA_OK);
return finishpcall(L, (status == LUA_YIELD), isx);
}
static int luaB_pcall (lua_State *L) { static int luaB_pcall (lua_State *L) {
int status; int status;
luaL_checkany(L, 1); luaL_checkany(L, 1);
status = lua_pcallk(L, lua_gettop(L) - 1, LUA_MULTRET, 0, 0, pcallcont); status = lua_pcallk(L, lua_gettop(L) - 1, LUA_MULTRET, 0, 0, pcallcont);
luaL_checkstack(L, 1, NULL); return finishpcall(L, (status == LUA_OK), 0);
lua_pushboolean(L, (status == LUA_OK));
lua_insert(L, 1);
return lua_gettop(L); /* return status + all results */
} }
@ -423,10 +431,7 @@ static int luaB_xpcall (lua_State *L) {
lua_copy(L, 2, 1); /* ...and error handler */ lua_copy(L, 2, 1); /* ...and error handler */
lua_replace(L, 2); lua_replace(L, 2);
status = lua_pcallk(L, n - 2, LUA_MULTRET, 1, 1, pcallcont); status = lua_pcallk(L, n - 2, LUA_MULTRET, 1, 1, pcallcont);
luaL_checkstack(L, 1, NULL); return finishpcall(L, (status == LUA_OK), 1);
lua_pushboolean(L, (status == LUA_OK));
lua_replace(L, 1);
return lua_gettop(L); /* return status + all results */
} }