diff options
author | Alexander Belopolsky <alexander.belopolsky@gmail.com> | 2012-09-09 17:16:15 (GMT) |
---|---|---|
committer | Alexander Belopolsky <alexander.belopolsky@gmail.com> | 2012-09-09 17:16:15 (GMT) |
commit | 59fb38b5820e9f61f6e64e506c2dad98f7f32971 (patch) | |
tree | 38aee2811ce37ba9acbb2493deb0c6334dd9ddfb /Lib/multiprocessing | |
parent | 47bcfff26be59c3979778b4621cd61f132c1a6d9 (diff) | |
download | cpython-59fb38b5820e9f61f6e64e506c2dad98f7f32971.zip cpython-59fb38b5820e9f61f6e64e506c2dad98f7f32971.tar.gz cpython-59fb38b5820e9f61f6e64e506c2dad98f7f32971.tar.bz2 |
Issue #15881: Fixed atexit hook in multiprocessing.
Diffstat (limited to 'Lib/multiprocessing')
-rw-r--r-- | Lib/multiprocessing/util.py | 54 |
1 files changed, 38 insertions, 16 deletions
diff --git a/Lib/multiprocessing/util.py b/Lib/multiprocessing/util.py index 30b7a85..68d6c1f 100644 --- a/Lib/multiprocessing/util.py +++ b/Lib/multiprocessing/util.py @@ -247,6 +247,12 @@ def _run_finalizers(minpriority=None): Finalizers with highest priority are called first; finalizers with the same priority will be called in reverse order of creation. ''' + if _finalizer_registry is None: + # This function may be called after this module's globals are + # destroyed. See the _exit_function function in this module for more + # notes. + return + if minpriority is None: f = lambda p : p[0][0] is not None else: @@ -278,24 +284,40 @@ def is_exiting(): _exiting = False -def _exit_function(): - global _exiting - - info('process shutting down') - debug('running all "atexit" finalizers with priority >= 0') - _run_finalizers(0) +def _exit_function(info=info, debug=debug, _run_finalizers=_run_finalizers, + active_children=active_children, + current_process=current_process): + # We hold on to references to functions in the arglist due to the + # situation described below, where this function is called after this + # module's globals are destroyed. - for p in active_children(): - if p._daemonic: - info('calling terminate() for daemon %s', p.name) - p._popen.terminate() - - for p in active_children(): - info('calling join() for process %s', p.name) - p.join() + global _exiting - debug('running the remaining "atexit" finalizers') - _run_finalizers() + if not _exiting: + info('process shutting down') + debug('running all "atexit" finalizers with priority >= 0') + _run_finalizers(0) + if current_process() is not None: + # We check if the current process is None here because if + # it's None, any call to ``active_children()`` will throw an + # AttributeError (active_children winds up trying to get + # attributes from util._current_process). This happens in a + # variety of shutdown circumstances that are not well-understood + # because module-scope variables are not apparently supposed to + # be destroyed until after this function is called. However, + # they are indeed destroyed before this function is called. See + # issues #9775 and #15881. Also related: #4106, #9205, and #9207. + for p in active_children(): + if p._daemonic: + info('calling terminate() for daemon %s', p.name) + p._popen.terminate() + + for p in active_children(): + info('calling join() for process %s', p.name) + p.join() + + debug('running the remaining "atexit" finalizers') + _run_finalizers() atexit.register(_exit_function) |