Skip to content

Commit

Permalink
[3.10] bpo-45021: Fix a hang in forked children (GH-28007) (GH-28480)
Browse files Browse the repository at this point in the history
_global_shutdown_lock should be reinitialized in forked children
(cherry picked from commit 0bfa110)


Co-authored-by: nullptr <3621629+0x0L@users.noreply.github.com>

Automerge-Triggered-By: GH:gpshead
  • Loading branch information
miss-islington committed Sep 20, 2021
1 parent 74c6acc commit d0d83a9
Show file tree
Hide file tree
Showing 3 changed files with 21 additions and 0 deletions.
6 changes: 6 additions & 0 deletions Lib/concurrent/futures/thread.py
Expand Up @@ -36,6 +36,12 @@ def _python_exit():
# See bpo-39812 for context.
threading._register_atexit(_python_exit)

# At fork, reinitialize the `_global_shutdown_lock` lock in the child process
if hasattr(os, 'register_at_fork'):
os.register_at_fork(before=_global_shutdown_lock.acquire,
after_in_child=_global_shutdown_lock._at_fork_reinit,
after_in_parent=_global_shutdown_lock.release)


class _WorkItem(object):
def __init__(self, future, fn, args, kwargs):
Expand Down
14 changes: 14 additions & 0 deletions Lib/test/test_concurrent_futures.py
Expand Up @@ -911,6 +911,20 @@ def test_idle_thread_reuse(self):
self.assertEqual(len(executor._threads), 1)
executor.shutdown(wait=True)

@unittest.skipUnless(hasattr(os, 'register_at_fork'), 'need os.register_at_fork')
def test_hang_global_shutdown_lock(self):
# bpo-45021: _global_shutdown_lock should be reinitialized in the child
# process, otherwise it will never exit
def submit(pool):
pool.submit(submit, pool)

with futures.ThreadPoolExecutor(1) as pool:
pool.submit(submit, pool)

for _ in range(50):
with futures.ProcessPoolExecutor(1, mp_context=get_context('fork')) as workers:
workers.submit(tuple)


class ProcessPoolExecutorTest(ExecutorTest):

Expand Down
@@ -0,0 +1 @@
Fix a potential deadlock at shutdown of forked children when using :mod:`concurrent.futures` module

0 comments on commit d0d83a9

Please sign in to comment.