The main thread cannot be closed

No thread started with pcall (instead of resume) can be closed,
because coroutine.close would not respect the expected number of
results from the protected call.
This commit is contained in:
Roberto Ierusalimschy 2025-06-13 14:08:38 -03:00
parent fd897027f1
commit e657a48ea5
3 changed files with 10 additions and 2 deletions

View file

@ -190,6 +190,9 @@ static int luaB_close (lua_State *L) {
} }
} }
case COS_RUN: /* running coroutine? */ case COS_RUN: /* running coroutine? */
lua_geti(L, LUA_REGISTRYINDEX, LUA_RIDX_MAINTHREAD); /* get main */
if (lua_tothread(L, -1) == co)
return luaL_error(L, "cannot close main thread");
lua_closethread(co, L); /* close itself */ lua_closethread(co, L); /* close itself */
lua_assert(0); /* previous call does not return */ lua_assert(0); /* previous call does not return */
return 0; return 0;

View file

@ -3284,8 +3284,8 @@ If @id{L} is equal to @id{from},
it corresponds to a thread closing itself. it corresponds to a thread closing itself.
In that case, In that case,
the call does not return; the call does not return;
instead, the resume or the protected call instead, the resume that (re)started the thread returns.
that (re)started the thread returns. The thread must be running inside a resume.
} }

View file

@ -158,6 +158,11 @@ do
local main = coroutine.running() local main = coroutine.running()
-- cannot close 'main'
local st, msg = pcall(coroutine.close, main);
assert(not st and string.find(msg, "main"))
-- cannot close a "normal" coroutine -- cannot close a "normal" coroutine
;(coroutine.wrap(function () ;(coroutine.wrap(function ()
local st, msg = pcall(coroutine.close, main) local st, msg = pcall(coroutine.close, main)