Issue #9573: os.fork now works when triggered as a side effect of import (the wisdom of actually relying on this remains questionable!)

This commit is contained in:
Nick Coghlan 2010-12-02 04:11:46 +00:00
parent d2bb830edc
commit b2ddf7979d
3 changed files with 57 additions and 5 deletions

View File

@ -8,13 +8,14 @@ import sys
import time import time
from test.fork_wait import ForkWait from test.fork_wait import ForkWait
from test.support import run_unittest, reap_children, get_attribute, import_module from test.support import (run_unittest, reap_children, get_attribute,
import_module, verbose)
threading = import_module('threading') threading = import_module('threading')
# Skip test if fork does not exist. # Skip test if fork does not exist.
get_attribute(os, 'fork') get_attribute(os, 'fork')
class ForkTest(ForkWait): class ForkTest(ForkWait):
def wait_impl(self, cpid): def wait_impl(self, cpid):
for i in range(10): for i in range(10):
@ -28,7 +29,8 @@ class ForkTest(ForkWait):
self.assertEqual(spid, cpid) self.assertEqual(spid, cpid)
self.assertEqual(status, 0, "cause = %d, exit = %d" % (status&0xff, status>>8)) self.assertEqual(status, 0, "cause = %d, exit = %d" % (status&0xff, status>>8))
def test_import_lock_fork(self): def test_threaded_import_lock_fork(self):
"""Check fork() in main thread works while a subthread is doing an import"""
import_started = threading.Event() import_started = threading.Event()
fake_module_name = "fake test module" fake_module_name = "fake test module"
partial_module = "partial" partial_module = "partial"
@ -45,11 +47,16 @@ class ForkTest(ForkWait):
import_started.wait() import_started.wait()
pid = os.fork() pid = os.fork()
try: try:
# PyOS_BeforeFork should have waited for the import to complete
# before forking, so the child can recreate the import lock
# correctly, but also won't see a partially initialised module
if not pid: if not pid:
m = __import__(fake_module_name) m = __import__(fake_module_name)
if m == complete_module: if m == complete_module:
os._exit(0) os._exit(0)
else: else:
if verbose > 1:
print("Child encountered partial module")
os._exit(1) os._exit(1)
else: else:
t.join() t.join()
@ -63,6 +70,39 @@ class ForkTest(ForkWait):
except OSError: except OSError:
pass pass
def test_nested_import_lock_fork(self):
"""Check fork() in main thread works while the main thread is doing an import"""
# Issue 9573: this used to trigger RuntimeError in the child process
def fork_with_import_lock(level):
release = 0
in_child = False
try:
try:
for i in range(level):
imp.acquire_lock()
release += 1
pid = os.fork()
in_child = not pid
finally:
for i in range(release):
imp.release_lock()
except RuntimeError:
if in_child:
if verbose > 1:
print("RuntimeError in child")
os._exit(1)
raise
if in_child:
os._exit(0)
self.wait_impl(pid)
# Check this works with various levels of nested
# import in the main thread
for level in range(5):
fork_with_import_lock(level)
def test_main(): def test_main():
run_unittest(ForkTest) run_unittest(ForkTest)
reap_children() reap_children()

View File

@ -46,6 +46,9 @@ Core and Builtins
Library Library
------- -------
- Issue #9573: os.fork() now works correctly when triggered as a side effect
of a module import
- Issue #10464: netrc now correctly handles lines with embedded '#' characters. - Issue #10464: netrc now correctly handles lines with embedded '#' characters.
- Added itertools.accumulate(). - Added itertools.accumulate().

View File

@ -325,9 +325,18 @@ _PyImport_ReInitLock(void)
{ {
if (import_lock != NULL) if (import_lock != NULL)
import_lock = PyThread_allocate_lock(); import_lock = PyThread_allocate_lock();
if (import_lock_level > 1) {
/* Forked as a side effect of import */
long me = PyThread_get_thread_ident();
PyThread_acquire_lock(import_lock, 0);
/* XXX: can the previous line fail? */
import_lock_thread = me;
import_lock_level--;
} else {
import_lock_thread = -1; import_lock_thread = -1;
import_lock_level = 0; import_lock_level = 0;
} }
}
#endif #endif