summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorGregory P. Smith <greg@krypto.org>2012-09-29 18:55:16 (GMT)
committerGregory P. Smith <greg@krypto.org>2012-09-29 18:55:16 (GMT)
commit288d9aec4b7df78cebf47dc8c12b7a93cc1a0faa (patch)
tree6875caa4aac01f9b8de98797dffd7ee50184fe87
parent1ed3ffebb9654622dc1089808ff95fadb84eda25 (diff)
parent02dee1acd8d905a2348e5ed4d9f97cb490e5e255 (diff)
downloadcpython-288d9aec4b7df78cebf47dc8c12b7a93cc1a0faa.zip
cpython-288d9aec4b7df78cebf47dc8c12b7a93cc1a0faa.tar.gz
cpython-288d9aec4b7df78cebf47dc8c12b7a93cc1a0faa.tar.bz2
Fixes issue #15756: subprocess.poll() now properly handles errno.ECHILD to
return a returncode of 0 when the child has already exited or cannot be waited on.
-rw-r--r--Lib/subprocess.py9
-rw-r--r--Lib/test/subprocessdata/sigchild_ignore.py11
-rw-r--r--Misc/NEWS4
3 files changed, 22 insertions, 2 deletions
diff --git a/Lib/subprocess.py b/Lib/subprocess.py
index cec1a24..775db50 100644
--- a/Lib/subprocess.py
+++ b/Lib/subprocess.py
@@ -1445,9 +1445,16 @@ class Popen(object):
pid, sts = _waitpid(self.pid, _WNOHANG)
if pid == self.pid:
self._handle_exitstatus(sts)
- except _os_error:
+ except _os_error as e:
if _deadstate is not None:
self.returncode = _deadstate
+ elif e.errno == errno.ECHILD:
+ # This happens if SIGCLD is set to be ignored or
+ # waiting for child processes has otherwise been
+ # disabled for our process. This child is dead, we
+ # can't get the status.
+ # http://bugs.python.org/issue15756
+ self.returncode = 0
return self.returncode
diff --git a/Lib/test/subprocessdata/sigchild_ignore.py b/Lib/test/subprocessdata/sigchild_ignore.py
index 6072aec..86320fb 100644
--- a/Lib/test/subprocessdata/sigchild_ignore.py
+++ b/Lib/test/subprocessdata/sigchild_ignore.py
@@ -1,6 +1,15 @@
-import signal, subprocess, sys
+import signal, subprocess, sys, time
# On Linux this causes os.waitpid to fail with OSError as the OS has already
# reaped our child process. The wait() passing the OSError on to the caller
# and causing us to exit with an error is what we are testing against.
signal.signal(signal.SIGCHLD, signal.SIG_IGN)
subprocess.Popen([sys.executable, '-c', 'print("albatross")']).wait()
+# Also ensure poll() handles an errno.ECHILD appropriately.
+p = subprocess.Popen([sys.executable, '-c', 'print("albatross")'])
+num_polls = 0
+while p.poll() is None:
+ # Waiting for the process to finish.
+ time.sleep(0.01) # Avoid being a CPU busy loop.
+ num_polls += 1
+ if num_polls > 3000:
+ raise RuntimeError('poll should have returned 0 within 30 seconds')
diff --git a/Misc/NEWS b/Misc/NEWS
index 248b79c..0012f1e 100644
--- a/Misc/NEWS
+++ b/Misc/NEWS
@@ -15,6 +15,10 @@ Core and Builtins
Library
-------
+- Issue #15756: subprocess.poll() now properly handles errno.ECHILD to
+ return a returncode of 0 when the child has already exited or cannot
+ be waited on.
+
Extension Modules
-----------------