blob: 7635eec148d88302a1ba9fd3a9afb36fb4e46753 [file] [log] [blame]
Victor Stinnerd6debb22017-03-27 16:05:26 +02001import os
Antoine Pitrouc08177a2017-06-28 23:29:29 +02002import random
Guido van Rossum4f17e3e1995-03-16 15:07:38 +00003import signal
Victor Stinner11517102014-07-29 23:31:34 +02004import socket
Antoine Pitrouf7d090c2017-06-29 16:40:14 +02005import statistics
Christian Heimes4fbc72b2008-03-22 00:47:35 +00006import subprocess
Victor Stinnerd6debb22017-03-27 16:05:26 +02007import sys
Antoine Pitroua6a4dc82017-09-07 18:56:24 +02008import threading
Victor Stinnerd6debb22017-03-27 16:05:26 +02009import time
10import unittest
11from test import support
Berker Peksagce643912015-05-06 06:33:17 +030012from test.support.script_helper import assert_python_ok, spawn_python
Victor Stinnerb3e72192011-05-08 01:46:11 +020013try:
Victor Stinner56e8c292014-07-21 12:30:22 +020014 import _testcapi
15except ImportError:
16 _testcapi = None
Christian Heimesc06950e2008-02-28 21:17:00 +000017
Guido van Rossumcc5a91d1997-04-16 00:29:15 +000018
Giampaolo Rodola'e09fb712014-04-04 15:34:17 +020019class GenericTests(unittest.TestCase):
20
21 def test_enums(self):
22 for name in dir(signal):
23 sig = getattr(signal, name)
24 if name in {'SIG_DFL', 'SIG_IGN'}:
25 self.assertIsInstance(sig, signal.Handlers)
26 elif name in {'SIG_BLOCK', 'SIG_UNBLOCK', 'SIG_SETMASK'}:
27 self.assertIsInstance(sig, signal.Sigmasks)
28 elif name.startswith('SIG') and not name.startswith('SIG_'):
29 self.assertIsInstance(sig, signal.Signals)
30 elif name.startswith('CTRL_'):
31 self.assertIsInstance(sig, signal.Signals)
32 self.assertEqual(sys.platform, "win32")
33
34
Brian Curtin3f004b12010-08-06 19:34:52 +000035@unittest.skipIf(sys.platform == "win32", "Not valid on Windows")
Victor Stinnerb3e72192011-05-08 01:46:11 +020036class PosixTests(unittest.TestCase):
Christian Heimes4fbc72b2008-03-22 00:47:35 +000037 def trivial_signal_handler(self, *args):
38 pass
39
Thomas Woutersed03b412007-08-28 21:37:11 +000040 def test_out_of_range_signal_number_raises_error(self):
41 self.assertRaises(ValueError, signal.getsignal, 4242)
42
Thomas Woutersed03b412007-08-28 21:37:11 +000043 self.assertRaises(ValueError, signal.signal, 4242,
Christian Heimes4fbc72b2008-03-22 00:47:35 +000044 self.trivial_signal_handler)
Thomas Woutersed03b412007-08-28 21:37:11 +000045
Antoine Pietri5d2a27d2018-03-12 14:42:34 +010046 self.assertRaises(ValueError, signal.strsignal, 4242)
47
Thomas Woutersed03b412007-08-28 21:37:11 +000048 def test_setting_signal_handler_to_none_raises_error(self):
49 self.assertRaises(TypeError, signal.signal,
50 signal.SIGUSR1, None)
51
Christian Heimes4fbc72b2008-03-22 00:47:35 +000052 def test_getsignal(self):
53 hup = signal.signal(signal.SIGHUP, self.trivial_signal_handler)
Giampaolo Rodola'e09fb712014-04-04 15:34:17 +020054 self.assertIsInstance(hup, signal.Handlers)
Ezio Melotti19f2aeb2010-11-21 01:30:29 +000055 self.assertEqual(signal.getsignal(signal.SIGHUP),
56 self.trivial_signal_handler)
Christian Heimes4fbc72b2008-03-22 00:47:35 +000057 signal.signal(signal.SIGHUP, hup)
Ezio Melotti19f2aeb2010-11-21 01:30:29 +000058 self.assertEqual(signal.getsignal(signal.SIGHUP), hup)
Christian Heimes4fbc72b2008-03-22 00:47:35 +000059
Antoine Pietri5d2a27d2018-03-12 14:42:34 +010060 def test_strsignal(self):
Antoine Pietri019f5b32018-03-12 20:03:14 +010061 self.assertIn("Interrupt", signal.strsignal(signal.SIGINT))
62 self.assertIn("Terminated", signal.strsignal(signal.SIGTERM))
Antoine Pietri5d2a27d2018-03-12 14:42:34 +010063
Victor Stinner32eb8402016-03-15 11:12:35 +010064 # Issue 3864, unknown if this affects earlier versions of freebsd also
Victor Stinner32eb8402016-03-15 11:12:35 +010065 def test_interprocess_signal(self):
66 dirname = os.path.dirname(__file__)
67 script = os.path.join(dirname, 'signalinterproctester.py')
68 assert_python_ok(script)
69
Christian Heimes4fbc72b2008-03-22 00:47:35 +000070
Brian Curtin3f004b12010-08-06 19:34:52 +000071@unittest.skipUnless(sys.platform == "win32", "Windows specific")
72class WindowsSignalTests(unittest.TestCase):
73 def test_issue9324(self):
Brian Curtineccd4d92010-10-01 15:09:53 +000074 # Updated for issue #10003, adding SIGBREAK
Brian Curtin3f004b12010-08-06 19:34:52 +000075 handler = lambda x, y: None
Nick Coghlan60b3ac72013-08-03 22:56:30 +100076 checked = set()
Brian Curtineccd4d92010-10-01 15:09:53 +000077 for sig in (signal.SIGABRT, signal.SIGBREAK, signal.SIGFPE,
78 signal.SIGILL, signal.SIGINT, signal.SIGSEGV,
79 signal.SIGTERM):
Nick Coghlan60b3ac72013-08-03 22:56:30 +100080 # Set and then reset a handler for signals that work on windows.
81 # Issue #18396, only for signals without a C-level handler.
82 if signal.getsignal(sig) is not None:
83 signal.signal(sig, signal.signal(sig, handler))
84 checked.add(sig)
85 # Issue #18396: Ensure the above loop at least tested *something*
86 self.assertTrue(checked)
Brian Curtin3f004b12010-08-06 19:34:52 +000087
88 with self.assertRaises(ValueError):
89 signal.signal(-1, handler)
Brian Curtin80cd4bf2010-08-07 03:52:38 +000090
91 with self.assertRaises(ValueError):
92 signal.signal(7, handler)
Brian Curtin3f004b12010-08-06 19:34:52 +000093
94
Benjamin Petersonc68a4a02013-01-18 00:10:24 -050095class WakeupFDTests(unittest.TestCase):
96
Nathaniel J. Smith902ab802017-12-17 20:10:18 -080097 def test_invalid_call(self):
98 # First parameter is positional-only
99 with self.assertRaises(TypeError):
100 signal.set_wakeup_fd(signum=signal.SIGINT)
101
102 # warn_on_full_buffer is a keyword-only parameter
103 with self.assertRaises(TypeError):
104 signal.set_wakeup_fd(signal.SIGINT, False)
105
Benjamin Petersonc68a4a02013-01-18 00:10:24 -0500106 def test_invalid_fd(self):
Victor Stinner1d8948e2014-07-24 22:51:05 +0200107 fd = support.make_bad_fd()
Victor Stinnera7d03d92014-07-21 17:17:28 +0200108 self.assertRaises((ValueError, OSError),
109 signal.set_wakeup_fd, fd)
Benjamin Petersonc68a4a02013-01-18 00:10:24 -0500110
Victor Stinner11517102014-07-29 23:31:34 +0200111 def test_invalid_socket(self):
112 sock = socket.socket()
113 fd = sock.fileno()
114 sock.close()
115 self.assertRaises((ValueError, OSError),
116 signal.set_wakeup_fd, fd)
117
Victor Stinnerd18ccd12014-07-24 21:58:53 +0200118 def test_set_wakeup_fd_result(self):
Victor Stinner1d8948e2014-07-24 22:51:05 +0200119 r1, w1 = os.pipe()
120 self.addCleanup(os.close, r1)
121 self.addCleanup(os.close, w1)
122 r2, w2 = os.pipe()
123 self.addCleanup(os.close, r2)
124 self.addCleanup(os.close, w2)
Victor Stinnerd18ccd12014-07-24 21:58:53 +0200125
Victor Stinner7cea44d2014-08-27 14:02:36 +0200126 if hasattr(os, 'set_blocking'):
127 os.set_blocking(w1, False)
128 os.set_blocking(w2, False)
Victor Stinner38227602014-08-27 12:59:44 +0200129
Victor Stinner1d8948e2014-07-24 22:51:05 +0200130 signal.set_wakeup_fd(w1)
Victor Stinnerc82a1792014-07-24 22:55:12 +0200131 self.assertEqual(signal.set_wakeup_fd(w2), w1)
132 self.assertEqual(signal.set_wakeup_fd(-1), w2)
133 self.assertEqual(signal.set_wakeup_fd(-1), -1)
Victor Stinner56e8c292014-07-21 12:30:22 +0200134
Victor Stinner11517102014-07-29 23:31:34 +0200135 def test_set_wakeup_fd_socket_result(self):
136 sock1 = socket.socket()
137 self.addCleanup(sock1.close)
Victor Stinner38227602014-08-27 12:59:44 +0200138 sock1.setblocking(False)
Victor Stinner11517102014-07-29 23:31:34 +0200139 fd1 = sock1.fileno()
140
141 sock2 = socket.socket()
142 self.addCleanup(sock2.close)
Victor Stinner38227602014-08-27 12:59:44 +0200143 sock2.setblocking(False)
Victor Stinner11517102014-07-29 23:31:34 +0200144 fd2 = sock2.fileno()
145
146 signal.set_wakeup_fd(fd1)
Victor Stinnerda565a72014-07-30 10:03:03 +0200147 self.assertEqual(signal.set_wakeup_fd(fd2), fd1)
148 self.assertEqual(signal.set_wakeup_fd(-1), fd2)
149 self.assertEqual(signal.set_wakeup_fd(-1), -1)
Victor Stinner11517102014-07-29 23:31:34 +0200150
Victor Stinner38227602014-08-27 12:59:44 +0200151 # On Windows, files are always blocking and Windows does not provide a
152 # function to test if a socket is in non-blocking mode.
153 @unittest.skipIf(sys.platform == "win32", "tests specific to POSIX")
154 def test_set_wakeup_fd_blocking(self):
155 rfd, wfd = os.pipe()
156 self.addCleanup(os.close, rfd)
157 self.addCleanup(os.close, wfd)
158
159 # fd must be non-blocking
160 os.set_blocking(wfd, True)
161 with self.assertRaises(ValueError) as cm:
162 signal.set_wakeup_fd(wfd)
163 self.assertEqual(str(cm.exception),
164 "the fd %s must be in non-blocking mode" % wfd)
165
166 # non-blocking is ok
167 os.set_blocking(wfd, False)
168 signal.set_wakeup_fd(wfd)
169 signal.set_wakeup_fd(-1)
170
Benjamin Petersonc68a4a02013-01-18 00:10:24 -0500171
Brian Curtin3f004b12010-08-06 19:34:52 +0000172@unittest.skipIf(sys.platform == "win32", "Not valid on Windows")
Christian Heimes5fb7c2a2007-12-24 08:52:31 +0000173class WakeupSignalTests(unittest.TestCase):
Victor Stinner56e8c292014-07-21 12:30:22 +0200174 @unittest.skipIf(_testcapi is None, 'need _testcapi')
Charles-François Natali027f9a32011-10-02 18:36:05 +0200175 def check_wakeup(self, test_body, *signals, ordered=True):
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200176 # use a subprocess to have only one thread
Victor Stinnere40b3aa2011-07-04 17:35:10 +0200177 code = """if 1:
Victor Stinner56e8c292014-07-21 12:30:22 +0200178 import _testcapi
Victor Stinnere40b3aa2011-07-04 17:35:10 +0200179 import os
180 import signal
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200181 import struct
Christian Heimes5fb7c2a2007-12-24 08:52:31 +0000182
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200183 signals = {!r}
Victor Stinnerc13ef662011-05-25 02:35:58 +0200184
Victor Stinnere40b3aa2011-07-04 17:35:10 +0200185 def handler(signum, frame):
186 pass
187
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200188 def check_signum(signals):
189 data = os.read(read, len(signals)+1)
190 raised = struct.unpack('%uB' % len(data), data)
Charles-François Natali027f9a32011-10-02 18:36:05 +0200191 if not {!r}:
192 raised = set(raised)
193 signals = set(signals)
Victor Stinner0a01f132011-07-04 18:06:35 +0200194 if raised != signals:
195 raise Exception("%r != %r" % (raised, signals))
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200196
Victor Stinnere40b3aa2011-07-04 17:35:10 +0200197 {}
198
199 signal.signal(signal.SIGALRM, handler)
200 read, write = os.pipe()
Victor Stinner1db9e7b2014-07-29 22:32:47 +0200201 os.set_blocking(write, False)
Victor Stinnere40b3aa2011-07-04 17:35:10 +0200202 signal.set_wakeup_fd(write)
203
204 test()
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200205 check_signum(signals)
Victor Stinnere40b3aa2011-07-04 17:35:10 +0200206
207 os.close(read)
208 os.close(write)
Giampaolo Rodola'e09fb712014-04-04 15:34:17 +0200209 """.format(tuple(map(int, signals)), ordered, test_body)
Victor Stinnere40b3aa2011-07-04 17:35:10 +0200210
211 assert_python_ok('-c', code)
Victor Stinnerd49b1f12011-05-08 02:03:15 +0200212
Victor Stinner56e8c292014-07-21 12:30:22 +0200213 @unittest.skipIf(_testcapi is None, 'need _testcapi')
Antoine Pitrou6f6ec372013-08-17 20:27:56 +0200214 def test_wakeup_write_error(self):
215 # Issue #16105: write() errors in the C signal handler should not
216 # pass silently.
217 # Use a subprocess to have only one thread.
218 code = """if 1:
Victor Stinner56e8c292014-07-21 12:30:22 +0200219 import _testcapi
Antoine Pitrou6f6ec372013-08-17 20:27:56 +0200220 import errno
Antoine Pitrou6f6ec372013-08-17 20:27:56 +0200221 import os
222 import signal
223 import sys
Antoine Pitrou6f6ec372013-08-17 20:27:56 +0200224 from test.support import captured_stderr
225
226 def handler(signum, frame):
227 1/0
228
229 signal.signal(signal.SIGALRM, handler)
230 r, w = os.pipe()
Victor Stinner1db9e7b2014-07-29 22:32:47 +0200231 os.set_blocking(r, False)
Antoine Pitrou6f6ec372013-08-17 20:27:56 +0200232
233 # Set wakeup_fd a read-only file descriptor to trigger the error
234 signal.set_wakeup_fd(r)
235 try:
236 with captured_stderr() as err:
Victor Stinner56e8c292014-07-21 12:30:22 +0200237 _testcapi.raise_signal(signal.SIGALRM)
Antoine Pitrou6f6ec372013-08-17 20:27:56 +0200238 except ZeroDivisionError:
239 # An ignored exception should have been printed out on stderr
240 err = err.getvalue()
241 if ('Exception ignored when trying to write to the signal wakeup fd'
242 not in err):
243 raise AssertionError(err)
244 if ('OSError: [Errno %d]' % errno.EBADF) not in err:
245 raise AssertionError(err)
246 else:
247 raise AssertionError("ZeroDivisionError not raised")
Victor Stinner56e8c292014-07-21 12:30:22 +0200248
249 os.close(r)
250 os.close(w)
Antoine Pitrou6f6ec372013-08-17 20:27:56 +0200251 """
Antoine Pitrou8f0bdda2013-08-17 21:43:47 +0200252 r, w = os.pipe()
253 try:
254 os.write(r, b'x')
255 except OSError:
256 pass
257 else:
258 self.skipTest("OS doesn't report write() error on the read end of a pipe")
259 finally:
260 os.close(r)
261 os.close(w)
Antoine Pitrou6f6ec372013-08-17 20:27:56 +0200262
263 assert_python_ok('-c', code)
264
Christian Heimes5fb7c2a2007-12-24 08:52:31 +0000265 def test_wakeup_fd_early(self):
Victor Stinnere40b3aa2011-07-04 17:35:10 +0200266 self.check_wakeup("""def test():
267 import select
268 import time
Christian Heimes5fb7c2a2007-12-24 08:52:31 +0000269
Victor Stinnere40b3aa2011-07-04 17:35:10 +0200270 TIMEOUT_FULL = 10
271 TIMEOUT_HALF = 5
272
Victor Stinner749a6a82015-03-30 22:09:14 +0200273 class InterruptSelect(Exception):
274 pass
275
276 def handler(signum, frame):
277 raise InterruptSelect
278 signal.signal(signal.SIGALRM, handler)
279
Victor Stinnere40b3aa2011-07-04 17:35:10 +0200280 signal.alarm(1)
Victor Stinner79d68f92015-03-19 21:54:09 +0100281
Victor Stinnere40b3aa2011-07-04 17:35:10 +0200282 # We attempt to get a signal during the sleep,
283 # before select is called
Victor Stinner79d68f92015-03-19 21:54:09 +0100284 try:
285 select.select([], [], [], TIMEOUT_FULL)
Victor Stinner749a6a82015-03-30 22:09:14 +0200286 except InterruptSelect:
Victor Stinner79d68f92015-03-19 21:54:09 +0100287 pass
288 else:
289 raise Exception("select() was not interrupted")
290
291 before_time = time.monotonic()
Victor Stinnere40b3aa2011-07-04 17:35:10 +0200292 select.select([read], [], [], TIMEOUT_FULL)
Victor Stinner79d68f92015-03-19 21:54:09 +0100293 after_time = time.monotonic()
294 dt = after_time - before_time
Victor Stinner0a01f132011-07-04 18:06:35 +0200295 if dt >= TIMEOUT_HALF:
296 raise Exception("%s >= %s" % (dt, TIMEOUT_HALF))
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200297 """, signal.SIGALRM)
Christian Heimes5fb7c2a2007-12-24 08:52:31 +0000298
299 def test_wakeup_fd_during(self):
Victor Stinnere40b3aa2011-07-04 17:35:10 +0200300 self.check_wakeup("""def test():
301 import select
302 import time
Christian Heimes5fb7c2a2007-12-24 08:52:31 +0000303
Victor Stinnere40b3aa2011-07-04 17:35:10 +0200304 TIMEOUT_FULL = 10
305 TIMEOUT_HALF = 5
Christian Heimes5fb7c2a2007-12-24 08:52:31 +0000306
Victor Stinner749a6a82015-03-30 22:09:14 +0200307 class InterruptSelect(Exception):
308 pass
309
310 def handler(signum, frame):
311 raise InterruptSelect
312 signal.signal(signal.SIGALRM, handler)
313
Victor Stinnere40b3aa2011-07-04 17:35:10 +0200314 signal.alarm(1)
Victor Stinner79d68f92015-03-19 21:54:09 +0100315 before_time = time.monotonic()
Victor Stinnere40b3aa2011-07-04 17:35:10 +0200316 # We attempt to get a signal during the select call
317 try:
318 select.select([read], [], [], TIMEOUT_FULL)
Victor Stinner749a6a82015-03-30 22:09:14 +0200319 except InterruptSelect:
Victor Stinnere40b3aa2011-07-04 17:35:10 +0200320 pass
321 else:
Victor Stinner749a6a82015-03-30 22:09:14 +0200322 raise Exception("select() was not interrupted")
Victor Stinner79d68f92015-03-19 21:54:09 +0100323 after_time = time.monotonic()
Victor Stinnere40b3aa2011-07-04 17:35:10 +0200324 dt = after_time - before_time
Victor Stinner0a01f132011-07-04 18:06:35 +0200325 if dt >= TIMEOUT_HALF:
326 raise Exception("%s >= %s" % (dt, TIMEOUT_HALF))
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200327 """, signal.SIGALRM)
Victor Stinnerd49b1f12011-05-08 02:03:15 +0200328
329 def test_signum(self):
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200330 self.check_wakeup("""def test():
Victor Stinner56e8c292014-07-21 12:30:22 +0200331 import _testcapi
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200332 signal.signal(signal.SIGUSR1, handler)
Victor Stinner56e8c292014-07-21 12:30:22 +0200333 _testcapi.raise_signal(signal.SIGUSR1)
334 _testcapi.raise_signal(signal.SIGALRM)
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200335 """, signal.SIGUSR1, signal.SIGALRM)
Christian Heimes5fb7c2a2007-12-24 08:52:31 +0000336
Victor Stinnerc13ef662011-05-25 02:35:58 +0200337 @unittest.skipUnless(hasattr(signal, 'pthread_sigmask'),
338 'need signal.pthread_sigmask()')
Victor Stinnerc13ef662011-05-25 02:35:58 +0200339 def test_pending(self):
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200340 self.check_wakeup("""def test():
341 signum1 = signal.SIGUSR1
342 signum2 = signal.SIGUSR2
Victor Stinnerc13ef662011-05-25 02:35:58 +0200343
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200344 signal.signal(signum1, handler)
345 signal.signal(signum2, handler)
Victor Stinnerc13ef662011-05-25 02:35:58 +0200346
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200347 signal.pthread_sigmask(signal.SIG_BLOCK, (signum1, signum2))
Victor Stinner56e8c292014-07-21 12:30:22 +0200348 _testcapi.raise_signal(signum1)
349 _testcapi.raise_signal(signum2)
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200350 # Unblocking the 2 signals calls the C signal handler twice
351 signal.pthread_sigmask(signal.SIG_UNBLOCK, (signum1, signum2))
Charles-François Natali027f9a32011-10-02 18:36:05 +0200352 """, signal.SIGUSR1, signal.SIGUSR2, ordered=False)
Victor Stinnerc13ef662011-05-25 02:35:58 +0200353
Christian Heimes5fb7c2a2007-12-24 08:52:31 +0000354
Victor Stinner11517102014-07-29 23:31:34 +0200355@unittest.skipUnless(hasattr(socket, 'socketpair'), 'need socket.socketpair')
356class WakeupSocketSignalTests(unittest.TestCase):
357
358 @unittest.skipIf(_testcapi is None, 'need _testcapi')
359 def test_socket(self):
360 # use a subprocess to have only one thread
361 code = """if 1:
362 import signal
363 import socket
364 import struct
365 import _testcapi
366
367 signum = signal.SIGINT
368 signals = (signum,)
369
370 def handler(signum, frame):
371 pass
372
373 signal.signal(signum, handler)
374
375 read, write = socket.socketpair()
376 read.setblocking(False)
377 write.setblocking(False)
378 signal.set_wakeup_fd(write.fileno())
379
380 _testcapi.raise_signal(signum)
381
382 data = read.recv(1)
383 if not data:
384 raise Exception("no signum written")
385 raised = struct.unpack('B', data)
386 if raised != signals:
387 raise Exception("%r != %r" % (raised, signals))
388
389 read.close()
390 write.close()
391 """
392
393 assert_python_ok('-c', code)
394
395 @unittest.skipIf(_testcapi is None, 'need _testcapi')
396 def test_send_error(self):
397 # Use a subprocess to have only one thread.
398 if os.name == 'nt':
399 action = 'send'
400 else:
401 action = 'write'
402 code = """if 1:
403 import errno
404 import signal
405 import socket
406 import sys
407 import time
408 import _testcapi
409 from test.support import captured_stderr
410
411 signum = signal.SIGINT
412
413 def handler(signum, frame):
414 pass
415
416 signal.signal(signum, handler)
417
418 read, write = socket.socketpair()
419 read.setblocking(False)
420 write.setblocking(False)
421
422 signal.set_wakeup_fd(write.fileno())
423
424 # Close sockets: send() will fail
425 read.close()
426 write.close()
427
428 with captured_stderr() as err:
429 _testcapi.raise_signal(signum)
430
431 err = err.getvalue()
432 if ('Exception ignored when trying to {action} to the signal wakeup fd'
433 not in err):
434 raise AssertionError(err)
435 """.format(action=action)
436 assert_python_ok('-c', code)
437
Nathaniel J. Smith902ab802017-12-17 20:10:18 -0800438 @unittest.skipIf(_testcapi is None, 'need _testcapi')
439 def test_warn_on_full_buffer(self):
440 # Use a subprocess to have only one thread.
441 if os.name == 'nt':
442 action = 'send'
443 else:
444 action = 'write'
445 code = """if 1:
446 import errno
447 import signal
448 import socket
449 import sys
450 import time
451 import _testcapi
452 from test.support import captured_stderr
453
454 signum = signal.SIGINT
455
456 # This handler will be called, but we intentionally won't read from
457 # the wakeup fd.
458 def handler(signum, frame):
459 pass
460
461 signal.signal(signum, handler)
462
463 read, write = socket.socketpair()
464 read.setblocking(False)
465 write.setblocking(False)
466
467 # Fill the send buffer
468 try:
469 while True:
470 write.send(b"x")
471 except BlockingIOError:
472 pass
473
474 # By default, we get a warning when a signal arrives
475 signal.set_wakeup_fd(write.fileno())
476
477 with captured_stderr() as err:
478 _testcapi.raise_signal(signum)
479
480 err = err.getvalue()
481 if ('Exception ignored when trying to {action} to the signal wakeup fd'
482 not in err):
483 raise AssertionError(err)
484
485 # And also if warn_on_full_buffer=True
486 signal.set_wakeup_fd(write.fileno(), warn_on_full_buffer=True)
487
488 with captured_stderr() as err:
489 _testcapi.raise_signal(signum)
490
491 err = err.getvalue()
492 if ('Exception ignored when trying to {action} to the signal wakeup fd'
493 not in err):
494 raise AssertionError(err)
495
496 # But not if warn_on_full_buffer=False
497 signal.set_wakeup_fd(write.fileno(), warn_on_full_buffer=False)
498
499 with captured_stderr() as err:
500 _testcapi.raise_signal(signum)
501
502 err = err.getvalue()
503 if err != "":
504 raise AssertionError("got unexpected output %r" % (err,))
505
506 # And then check the default again, to make sure warn_on_full_buffer
507 # settings don't leak across calls.
508 signal.set_wakeup_fd(write.fileno())
509
510 with captured_stderr() as err:
511 _testcapi.raise_signal(signum)
512
513 err = err.getvalue()
514 if ('Exception ignored when trying to {action} to the signal wakeup fd'
515 not in err):
516 raise AssertionError(err)
517
518 """.format(action=action)
519 assert_python_ok('-c', code)
520
Victor Stinner11517102014-07-29 23:31:34 +0200521
Brian Curtin3f004b12010-08-06 19:34:52 +0000522@unittest.skipIf(sys.platform == "win32", "Not valid on Windows")
Christian Heimes8640e742008-02-23 16:23:06 +0000523class SiginterruptTest(unittest.TestCase):
Jean-Paul Calderone9c39bc72010-05-09 03:25:16 +0000524
Victor Stinnerd6284962011-06-20 23:28:09 +0200525 def readpipe_interrupted(self, interrupt):
Jean-Paul Calderone9c39bc72010-05-09 03:25:16 +0000526 """Perform a read during which a signal will arrive. Return True if the
527 read is interrupted by the signal and raises an exception. Return False
528 if it returns normally.
529 """
Victor Stinnerd6284962011-06-20 23:28:09 +0200530 # use a subprocess to have only one thread, to have a timeout on the
531 # blocking read and to not touch signal handling in this process
532 code = """if 1:
533 import errno
534 import os
535 import signal
536 import sys
Jean-Paul Calderone9c39bc72010-05-09 03:25:16 +0000537
Victor Stinnerd6284962011-06-20 23:28:09 +0200538 interrupt = %r
539 r, w = os.pipe()
Christian Heimes8640e742008-02-23 16:23:06 +0000540
Victor Stinnerd6284962011-06-20 23:28:09 +0200541 def handler(signum, frame):
Charles-François Natali6e6c59b2015-02-07 13:27:50 +0000542 1 / 0
Victor Stinnerd6284962011-06-20 23:28:09 +0200543
544 signal.signal(signal.SIGALRM, handler)
545 if interrupt is not None:
546 signal.siginterrupt(signal.SIGALRM, interrupt)
547
Victor Stinnerdfde0d42011-07-01 15:58:39 +0200548 print("ready")
549 sys.stdout.flush()
550
Victor Stinnerd6284962011-06-20 23:28:09 +0200551 # run the test twice
Victor Stinner56e8c292014-07-21 12:30:22 +0200552 try:
553 for loop in range(2):
554 # send a SIGALRM in a second (during the read)
555 signal.alarm(1)
556 try:
557 # blocking call: read from a pipe without data
558 os.read(r, 1)
Charles-François Natali6e6c59b2015-02-07 13:27:50 +0000559 except ZeroDivisionError:
560 pass
Victor Stinner56e8c292014-07-21 12:30:22 +0200561 else:
562 sys.exit(2)
563 sys.exit(3)
564 finally:
565 os.close(r)
566 os.close(w)
Victor Stinnerd6284962011-06-20 23:28:09 +0200567 """ % (interrupt,)
568 with spawn_python('-c', code) as process:
Christian Heimes8640e742008-02-23 16:23:06 +0000569 try:
Victor Stinner45273652011-06-22 22:15:51 +0200570 # wait until the child process is loaded and has started
571 first_line = process.stdout.readline()
572
Victor Stinner19e5bcd2011-07-01 15:59:54 +0200573 stdout, stderr = process.communicate(timeout=5.0)
Victor Stinnerd6284962011-06-20 23:28:09 +0200574 except subprocess.TimeoutExpired:
575 process.kill()
Christian Heimes8640e742008-02-23 16:23:06 +0000576 return False
Victor Stinnerd6284962011-06-20 23:28:09 +0200577 else:
Victor Stinner45273652011-06-22 22:15:51 +0200578 stdout = first_line + stdout
Victor Stinnerd6284962011-06-20 23:28:09 +0200579 exitcode = process.wait()
580 if exitcode not in (2, 3):
Antoine Pitroudab4e8a2014-05-11 19:05:23 +0200581 raise Exception("Child error (exit code %s): %r"
Victor Stinnerd6284962011-06-20 23:28:09 +0200582 % (exitcode, stdout))
583 return (exitcode == 3)
Christian Heimes8640e742008-02-23 16:23:06 +0000584
585 def test_without_siginterrupt(self):
Victor Stinner3a7f0f02011-05-08 02:10:36 +0200586 # If a signal handler is installed and siginterrupt is not called
587 # at all, when that signal arrives, it interrupts a syscall that's in
588 # progress.
Victor Stinnerd6284962011-06-20 23:28:09 +0200589 interrupted = self.readpipe_interrupted(None)
590 self.assertTrue(interrupted)
Christian Heimes8640e742008-02-23 16:23:06 +0000591
592 def test_siginterrupt_on(self):
Victor Stinner3a7f0f02011-05-08 02:10:36 +0200593 # If a signal handler is installed and siginterrupt is called with
594 # a true value for the second argument, when that signal arrives, it
595 # interrupts a syscall that's in progress.
Victor Stinnerd6284962011-06-20 23:28:09 +0200596 interrupted = self.readpipe_interrupted(True)
597 self.assertTrue(interrupted)
Christian Heimes8640e742008-02-23 16:23:06 +0000598
599 def test_siginterrupt_off(self):
Victor Stinner3a7f0f02011-05-08 02:10:36 +0200600 # If a signal handler is installed and siginterrupt is called with
601 # a false value for the second argument, when that signal arrives, it
602 # does not interrupt a syscall that's in progress.
Victor Stinnerd6284962011-06-20 23:28:09 +0200603 interrupted = self.readpipe_interrupted(False)
604 self.assertFalse(interrupted)
Jean-Paul Calderone9c39bc72010-05-09 03:25:16 +0000605
606
Brian Curtin3f004b12010-08-06 19:34:52 +0000607@unittest.skipIf(sys.platform == "win32", "Not valid on Windows")
Martin v. Löwis823725e2008-03-24 13:39:54 +0000608class ItimerTest(unittest.TestCase):
609 def setUp(self):
610 self.hndl_called = False
611 self.hndl_count = 0
612 self.itimer = None
Christian Heimescc47b052008-03-25 14:56:36 +0000613 self.old_alarm = signal.signal(signal.SIGALRM, self.sig_alrm)
Martin v. Löwis823725e2008-03-24 13:39:54 +0000614
615 def tearDown(self):
Christian Heimescc47b052008-03-25 14:56:36 +0000616 signal.signal(signal.SIGALRM, self.old_alarm)
Martin v. Löwis823725e2008-03-24 13:39:54 +0000617 if self.itimer is not None: # test_itimer_exc doesn't change this attr
618 # just ensure that itimer is stopped
619 signal.setitimer(self.itimer, 0)
620
621 def sig_alrm(self, *args):
622 self.hndl_called = True
Martin v. Löwis823725e2008-03-24 13:39:54 +0000623
624 def sig_vtalrm(self, *args):
625 self.hndl_called = True
626
627 if self.hndl_count > 3:
628 # it shouldn't be here, because it should have been disabled.
629 raise signal.ItimerError("setitimer didn't disable ITIMER_VIRTUAL "
630 "timer.")
631 elif self.hndl_count == 3:
632 # disable ITIMER_VIRTUAL, this function shouldn't be called anymore
633 signal.setitimer(signal.ITIMER_VIRTUAL, 0)
Martin v. Löwis823725e2008-03-24 13:39:54 +0000634
635 self.hndl_count += 1
636
Martin v. Löwis823725e2008-03-24 13:39:54 +0000637 def sig_prof(self, *args):
638 self.hndl_called = True
639 signal.setitimer(signal.ITIMER_PROF, 0)
640
Martin v. Löwis823725e2008-03-24 13:39:54 +0000641 def test_itimer_exc(self):
642 # XXX I'm assuming -1 is an invalid itimer, but maybe some platform
643 # defines it ?
644 self.assertRaises(signal.ItimerError, signal.setitimer, -1, 0)
Christian Heimescc47b052008-03-25 14:56:36 +0000645 # Negative times are treated as zero on some platforms.
646 if 0:
647 self.assertRaises(signal.ItimerError,
648 signal.setitimer, signal.ITIMER_REAL, -1)
Martin v. Löwis823725e2008-03-24 13:39:54 +0000649
650 def test_itimer_real(self):
651 self.itimer = signal.ITIMER_REAL
Martin v. Löwis823725e2008-03-24 13:39:54 +0000652 signal.setitimer(self.itimer, 1.0)
Martin v. Löwis823725e2008-03-24 13:39:54 +0000653 signal.pause()
Martin v. Löwis823725e2008-03-24 13:39:54 +0000654 self.assertEqual(self.hndl_called, True)
655
R. David Murray44546f82010-04-21 01:59:28 +0000656 # Issue 3864, unknown if this affects earlier versions of freebsd also
Victor Stinner13ff2452018-01-22 18:32:50 +0100657 @unittest.skipIf(sys.platform in ('netbsd5',),
Gregory P. Smith397cd8a2010-10-17 04:23:21 +0000658 'itimer not reliable (does not mix well with threading) on some BSDs.')
Martin v. Löwis823725e2008-03-24 13:39:54 +0000659 def test_itimer_virtual(self):
660 self.itimer = signal.ITIMER_VIRTUAL
661 signal.signal(signal.SIGVTALRM, self.sig_vtalrm)
662 signal.setitimer(self.itimer, 0.3, 0.2)
663
Victor Stinner79d68f92015-03-19 21:54:09 +0100664 start_time = time.monotonic()
665 while time.monotonic() - start_time < 60.0:
Mark Dickinson95078872009-10-04 18:47:48 +0000666 # use up some virtual time by doing real work
667 _ = pow(12345, 67890, 10000019)
Martin v. Löwis823725e2008-03-24 13:39:54 +0000668 if signal.getitimer(self.itimer) == (0.0, 0.0):
669 break # sig_vtalrm handler stopped this itimer
Stefan Krahf1da973042010-04-20 08:15:14 +0000670 else: # Issue 8424
Benjamin Peterson9b140f62010-05-15 18:00:56 +0000671 self.skipTest("timeout: likely cause: machine too slow or load too "
672 "high")
Martin v. Löwis823725e2008-03-24 13:39:54 +0000673
674 # virtual itimer should be (0.0, 0.0) now
Ezio Melotti19f2aeb2010-11-21 01:30:29 +0000675 self.assertEqual(signal.getitimer(self.itimer), (0.0, 0.0))
Martin v. Löwis823725e2008-03-24 13:39:54 +0000676 # and the handler should have been called
Ezio Melotti19f2aeb2010-11-21 01:30:29 +0000677 self.assertEqual(self.hndl_called, True)
Martin v. Löwis823725e2008-03-24 13:39:54 +0000678
679 def test_itimer_prof(self):
680 self.itimer = signal.ITIMER_PROF
681 signal.signal(signal.SIGPROF, self.sig_prof)
Neal Norwitzf5c7c2e2008-04-05 04:47:45 +0000682 signal.setitimer(self.itimer, 0.2, 0.2)
Martin v. Löwis823725e2008-03-24 13:39:54 +0000683
Victor Stinner79d68f92015-03-19 21:54:09 +0100684 start_time = time.monotonic()
685 while time.monotonic() - start_time < 60.0:
Mark Dickinson78373472009-10-31 10:39:21 +0000686 # do some work
687 _ = pow(12345, 67890, 10000019)
Martin v. Löwis823725e2008-03-24 13:39:54 +0000688 if signal.getitimer(self.itimer) == (0.0, 0.0):
689 break # sig_prof handler stopped this itimer
Stefan Krahf1da973042010-04-20 08:15:14 +0000690 else: # Issue 8424
Benjamin Peterson9b140f62010-05-15 18:00:56 +0000691 self.skipTest("timeout: likely cause: machine too slow or load too "
692 "high")
Martin v. Löwis823725e2008-03-24 13:39:54 +0000693
Neal Norwitzf5c7c2e2008-04-05 04:47:45 +0000694 # profiling itimer should be (0.0, 0.0) now
Ezio Melotti19f2aeb2010-11-21 01:30:29 +0000695 self.assertEqual(signal.getitimer(self.itimer), (0.0, 0.0))
Neal Norwitzf5c7c2e2008-04-05 04:47:45 +0000696 # and the handler should have been called
Martin v. Löwis823725e2008-03-24 13:39:54 +0000697 self.assertEqual(self.hndl_called, True)
698
Antoine Pitrou729780a2017-06-30 10:01:05 +0200699 def test_setitimer_tiny(self):
700 # bpo-30807: C setitimer() takes a microsecond-resolution interval.
701 # Check that float -> timeval conversion doesn't round
702 # the interval down to zero, which would disable the timer.
703 self.itimer = signal.ITIMER_REAL
704 signal.setitimer(self.itimer, 1e-6)
705 time.sleep(1)
706 self.assertEqual(self.hndl_called, True)
707
Victor Stinnera9293352011-04-30 15:21:58 +0200708
Victor Stinner35b300c2011-05-04 13:20:35 +0200709class PendingSignalsTests(unittest.TestCase):
710 """
Victor Stinnerb3e72192011-05-08 01:46:11 +0200711 Test pthread_sigmask(), pthread_kill(), sigpending() and sigwait()
712 functions.
Victor Stinner35b300c2011-05-04 13:20:35 +0200713 """
Victor Stinnerb3e72192011-05-08 01:46:11 +0200714 @unittest.skipUnless(hasattr(signal, 'sigpending'),
715 'need signal.sigpending()')
716 def test_sigpending_empty(self):
717 self.assertEqual(signal.sigpending(), set())
718
719 @unittest.skipUnless(hasattr(signal, 'pthread_sigmask'),
720 'need signal.pthread_sigmask()')
721 @unittest.skipUnless(hasattr(signal, 'sigpending'),
722 'need signal.sigpending()')
723 def test_sigpending(self):
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200724 code = """if 1:
725 import os
726 import signal
Victor Stinnerb3e72192011-05-08 01:46:11 +0200727
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200728 def handler(signum, frame):
729 1/0
Victor Stinnerb3e72192011-05-08 01:46:11 +0200730
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200731 signum = signal.SIGUSR1
732 signal.signal(signum, handler)
733
734 signal.pthread_sigmask(signal.SIG_BLOCK, [signum])
735 os.kill(os.getpid(), signum)
736 pending = signal.sigpending()
Giampaolo Rodola'e09fb712014-04-04 15:34:17 +0200737 for sig in pending:
738 assert isinstance(sig, signal.Signals), repr(pending)
Victor Stinner0a01f132011-07-04 18:06:35 +0200739 if pending != {signum}:
740 raise Exception('%s != {%s}' % (pending, signum))
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200741 try:
742 signal.pthread_sigmask(signal.SIG_UNBLOCK, [signum])
743 except ZeroDivisionError:
744 pass
745 else:
746 raise Exception("ZeroDivisionError not raised")
747 """
748 assert_python_ok('-c', code)
Victor Stinnerb3e72192011-05-08 01:46:11 +0200749
750 @unittest.skipUnless(hasattr(signal, 'pthread_kill'),
751 'need signal.pthread_kill()')
752 def test_pthread_kill(self):
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200753 code = """if 1:
754 import signal
755 import threading
756 import sys
Victor Stinnerb3e72192011-05-08 01:46:11 +0200757
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200758 signum = signal.SIGUSR1
Victor Stinnerb3e72192011-05-08 01:46:11 +0200759
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200760 def handler(signum, frame):
761 1/0
762
763 signal.signal(signum, handler)
764
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200765 tid = threading.get_ident()
766 try:
767 signal.pthread_kill(tid, signum)
768 except ZeroDivisionError:
769 pass
770 else:
771 raise Exception("ZeroDivisionError not raised")
772 """
773 assert_python_ok('-c', code)
Victor Stinnerb3e72192011-05-08 01:46:11 +0200774
Victor Stinner7f294d12011-06-10 14:02:10 +0200775 @unittest.skipUnless(hasattr(signal, 'pthread_sigmask'),
776 'need signal.pthread_sigmask()')
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200777 def wait_helper(self, blocked, test):
Victor Stinner9e8b82f2011-06-29 10:43:02 +0200778 """
779 test: body of the "def test(signum):" function.
780 blocked: number of the blocked signal
781 """
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200782 code = '''if 1:
783 import signal
784 import sys
Giampaolo Rodola'e09fb712014-04-04 15:34:17 +0200785 from signal import Signals
Victor Stinner9e8b82f2011-06-29 10:43:02 +0200786
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200787 def handler(signum, frame):
788 1/0
Victor Stinner9e8b82f2011-06-29 10:43:02 +0200789
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200790 %s
Victor Stinner9e8b82f2011-06-29 10:43:02 +0200791
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200792 blocked = %s
793 signum = signal.SIGALRM
Victor Stinner9e8b82f2011-06-29 10:43:02 +0200794
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200795 # child: block and wait the signal
796 try:
797 signal.signal(signum, handler)
798 signal.pthread_sigmask(signal.SIG_BLOCK, [blocked])
Victor Stinner9e8b82f2011-06-29 10:43:02 +0200799
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200800 # Do the tests
801 test(signum)
Victor Stinner9e8b82f2011-06-29 10:43:02 +0200802
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200803 # The handler must not be called on unblock
804 try:
805 signal.pthread_sigmask(signal.SIG_UNBLOCK, [blocked])
806 except ZeroDivisionError:
807 print("the signal handler has been called",
808 file=sys.stderr)
809 sys.exit(1)
810 except BaseException as err:
811 print("error: {}".format(err), file=sys.stderr)
812 sys.stderr.flush()
813 sys.exit(1)
814 ''' % (test.strip(), blocked)
Victor Stinner415007e2011-06-13 16:19:06 +0200815
Ross Lagerwallbc808222011-06-25 12:13:40 +0200816 # sig*wait* must be called with the signal blocked: since the current
Victor Stinner9e8b82f2011-06-29 10:43:02 +0200817 # process might have several threads running, use a subprocess to have
818 # a single thread.
819 assert_python_ok('-c', code)
Victor Stinneraf494602011-06-10 12:48:13 +0200820
Victor Stinnerb3e72192011-05-08 01:46:11 +0200821 @unittest.skipUnless(hasattr(signal, 'sigwait'),
822 'need signal.sigwait()')
Ross Lagerwallbc808222011-06-25 12:13:40 +0200823 def test_sigwait(self):
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200824 self.wait_helper(signal.SIGALRM, '''
825 def test(signum):
Ross Lagerwallbc808222011-06-25 12:13:40 +0200826 signal.alarm(1)
Victor Stinner9e8b82f2011-06-29 10:43:02 +0200827 received = signal.sigwait([signum])
Giampaolo Rodola'e09fb712014-04-04 15:34:17 +0200828 assert isinstance(received, signal.Signals), received
Victor Stinner0a01f132011-07-04 18:06:35 +0200829 if received != signum:
830 raise Exception('received %s, not %s' % (received, signum))
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200831 ''')
Ross Lagerwallbc808222011-06-25 12:13:40 +0200832
833 @unittest.skipUnless(hasattr(signal, 'sigwaitinfo'),
834 'need signal.sigwaitinfo()')
835 def test_sigwaitinfo(self):
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200836 self.wait_helper(signal.SIGALRM, '''
837 def test(signum):
Ross Lagerwallbc808222011-06-25 12:13:40 +0200838 signal.alarm(1)
839 info = signal.sigwaitinfo([signum])
Victor Stinner0a01f132011-07-04 18:06:35 +0200840 if info.si_signo != signum:
841 raise Exception("info.si_signo != %s" % signum)
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200842 ''')
Ross Lagerwallbc808222011-06-25 12:13:40 +0200843
844 @unittest.skipUnless(hasattr(signal, 'sigtimedwait'),
845 'need signal.sigtimedwait()')
846 def test_sigtimedwait(self):
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200847 self.wait_helper(signal.SIGALRM, '''
848 def test(signum):
Ross Lagerwallbc808222011-06-25 12:13:40 +0200849 signal.alarm(1)
Victor Stinner643cd682012-03-02 22:54:03 +0100850 info = signal.sigtimedwait([signum], 10.1000)
Victor Stinner0a01f132011-07-04 18:06:35 +0200851 if info.si_signo != signum:
852 raise Exception('info.si_signo != %s' % signum)
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200853 ''')
Ross Lagerwallbc808222011-06-25 12:13:40 +0200854
Ross Lagerwallbc808222011-06-25 12:13:40 +0200855 @unittest.skipUnless(hasattr(signal, 'sigtimedwait'),
856 'need signal.sigtimedwait()')
857 def test_sigtimedwait_poll(self):
Victor Stinner9e8b82f2011-06-29 10:43:02 +0200858 # check that polling with sigtimedwait works
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200859 self.wait_helper(signal.SIGALRM, '''
860 def test(signum):
Victor Stinner9e8b82f2011-06-29 10:43:02 +0200861 import os
862 os.kill(os.getpid(), signum)
Victor Stinner643cd682012-03-02 22:54:03 +0100863 info = signal.sigtimedwait([signum], 0)
Victor Stinner0a01f132011-07-04 18:06:35 +0200864 if info.si_signo != signum:
865 raise Exception('info.si_signo != %s' % signum)
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200866 ''')
Ross Lagerwallbc808222011-06-25 12:13:40 +0200867
868 @unittest.skipUnless(hasattr(signal, 'sigtimedwait'),
869 'need signal.sigtimedwait()')
870 def test_sigtimedwait_timeout(self):
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200871 self.wait_helper(signal.SIGALRM, '''
872 def test(signum):
Victor Stinner643cd682012-03-02 22:54:03 +0100873 received = signal.sigtimedwait([signum], 1.0)
Victor Stinner0a01f132011-07-04 18:06:35 +0200874 if received is not None:
875 raise Exception("received=%r" % (received,))
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200876 ''')
Ross Lagerwallbc808222011-06-25 12:13:40 +0200877
878 @unittest.skipUnless(hasattr(signal, 'sigtimedwait'),
879 'need signal.sigtimedwait()')
880 def test_sigtimedwait_negative_timeout(self):
881 signum = signal.SIGALRM
Victor Stinner643cd682012-03-02 22:54:03 +0100882 self.assertRaises(ValueError, signal.sigtimedwait, [signum], -1.0)
Ross Lagerwallbc808222011-06-25 12:13:40 +0200883
Ross Lagerwallbc808222011-06-25 12:13:40 +0200884 @unittest.skipUnless(hasattr(signal, 'sigwait'),
885 'need signal.sigwait()')
Victor Stinner415007e2011-06-13 16:19:06 +0200886 @unittest.skipUnless(hasattr(signal, 'pthread_sigmask'),
887 'need signal.pthread_sigmask()')
Victor Stinner10c30d62011-06-10 01:39:53 +0200888 def test_sigwait_thread(self):
Victor Stinner415007e2011-06-13 16:19:06 +0200889 # Check that calling sigwait() from a thread doesn't suspend the whole
890 # process. A new interpreter is spawned to avoid problems when mixing
891 # threads and fork(): only async-safe functions are allowed between
892 # fork() and exec().
893 assert_python_ok("-c", """if True:
894 import os, threading, sys, time, signal
Victor Stinner10c30d62011-06-10 01:39:53 +0200895
Victor Stinner415007e2011-06-13 16:19:06 +0200896 # the default handler terminates the process
897 signum = signal.SIGUSR1
898
899 def kill_later():
900 # wait until the main thread is waiting in sigwait()
901 time.sleep(1)
902 os.kill(os.getpid(), signum)
903
904 # the signal must be blocked by all the threads
905 signal.pthread_sigmask(signal.SIG_BLOCK, [signum])
906 killer = threading.Thread(target=kill_later)
Victor Stinneraf494602011-06-10 12:48:13 +0200907 killer.start()
908 received = signal.sigwait([signum])
909 if received != signum:
910 print("sigwait() received %s, not %s" % (received, signum),
911 file=sys.stderr)
Victor Stinner415007e2011-06-13 16:19:06 +0200912 sys.exit(1)
Victor Stinner10c30d62011-06-10 01:39:53 +0200913 killer.join()
Victor Stinner415007e2011-06-13 16:19:06 +0200914 # unblock the signal, which should have been cleared by sigwait()
915 signal.pthread_sigmask(signal.SIG_UNBLOCK, [signum])
916 """)
Victor Stinneraf494602011-06-10 12:48:13 +0200917
Victor Stinnerb3e72192011-05-08 01:46:11 +0200918 @unittest.skipUnless(hasattr(signal, 'pthread_sigmask'),
919 'need signal.pthread_sigmask()')
920 def test_pthread_sigmask_arguments(self):
921 self.assertRaises(TypeError, signal.pthread_sigmask)
922 self.assertRaises(TypeError, signal.pthread_sigmask, 1)
923 self.assertRaises(TypeError, signal.pthread_sigmask, 1, 2, 3)
924 self.assertRaises(OSError, signal.pthread_sigmask, 1700, [])
925
926 @unittest.skipUnless(hasattr(signal, 'pthread_sigmask'),
927 'need signal.pthread_sigmask()')
928 def test_pthread_sigmask(self):
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200929 code = """if 1:
930 import signal
931 import os; import threading
932
933 def handler(signum, frame):
934 1/0
935
936 def kill(signum):
937 os.kill(os.getpid(), signum)
938
Giampaolo Rodola'e09fb712014-04-04 15:34:17 +0200939 def check_mask(mask):
940 for sig in mask:
941 assert isinstance(sig, signal.Signals), repr(sig)
942
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200943 def read_sigmask():
Giampaolo Rodola'e09fb712014-04-04 15:34:17 +0200944 sigmask = signal.pthread_sigmask(signal.SIG_BLOCK, [])
945 check_mask(sigmask)
946 return sigmask
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200947
Victor Stinnerb3e72192011-05-08 01:46:11 +0200948 signum = signal.SIGUSR1
Victor Stinner6fd49e12011-05-04 12:38:03 +0200949
Victor Stinnerd0e516d2011-05-03 14:57:12 +0200950 # Install our signal handler
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200951 old_handler = signal.signal(signum, handler)
Victor Stinnera9293352011-04-30 15:21:58 +0200952
Victor Stinnerd0e516d2011-05-03 14:57:12 +0200953 # Unblock SIGUSR1 (and copy the old mask) to test our signal handler
Victor Stinnera9293352011-04-30 15:21:58 +0200954 old_mask = signal.pthread_sigmask(signal.SIG_UNBLOCK, [signum])
Giampaolo Rodola'e09fb712014-04-04 15:34:17 +0200955 check_mask(old_mask)
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200956 try:
957 kill(signum)
958 except ZeroDivisionError:
959 pass
960 else:
961 raise Exception("ZeroDivisionError not raised")
Victor Stinnera9293352011-04-30 15:21:58 +0200962
Victor Stinnerd0e516d2011-05-03 14:57:12 +0200963 # Block and then raise SIGUSR1. The signal is blocked: the signal
964 # handler is not called, and the signal is now pending
Giampaolo Rodola'e09fb712014-04-04 15:34:17 +0200965 mask = signal.pthread_sigmask(signal.SIG_BLOCK, [signum])
966 check_mask(mask)
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200967 kill(signum)
Victor Stinnera9293352011-04-30 15:21:58 +0200968
Victor Stinnerd0e516d2011-05-03 14:57:12 +0200969 # Check the new mask
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200970 blocked = read_sigmask()
Giampaolo Rodola'e09fb712014-04-04 15:34:17 +0200971 check_mask(blocked)
Victor Stinner0a01f132011-07-04 18:06:35 +0200972 if signum not in blocked:
973 raise Exception("%s not in %s" % (signum, blocked))
974 if old_mask ^ blocked != {signum}:
975 raise Exception("%s ^ %s != {%s}" % (old_mask, blocked, signum))
Victor Stinnera9293352011-04-30 15:21:58 +0200976
Victor Stinnerd0e516d2011-05-03 14:57:12 +0200977 # Unblock SIGUSR1
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200978 try:
R David Murrayfc069992013-12-13 20:52:19 -0500979 # unblock the pending signal calls immediately the signal handler
Victor Stinnerd0e516d2011-05-03 14:57:12 +0200980 signal.pthread_sigmask(signal.SIG_UNBLOCK, [signum])
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200981 except ZeroDivisionError:
982 pass
983 else:
984 raise Exception("ZeroDivisionError not raised")
985 try:
986 kill(signum)
987 except ZeroDivisionError:
988 pass
989 else:
990 raise Exception("ZeroDivisionError not raised")
Victor Stinnera9293352011-04-30 15:21:58 +0200991
Victor Stinnerd0e516d2011-05-03 14:57:12 +0200992 # Check the new mask
Victor Stinnerd554cdf2011-07-04 17:49:40 +0200993 unblocked = read_sigmask()
Victor Stinner0a01f132011-07-04 18:06:35 +0200994 if signum in unblocked:
995 raise Exception("%s in %s" % (signum, unblocked))
996 if blocked ^ unblocked != {signum}:
997 raise Exception("%s ^ %s != {%s}" % (blocked, unblocked, signum))
998 if old_mask != unblocked:
999 raise Exception("%s != %s" % (old_mask, unblocked))
Victor Stinnerd554cdf2011-07-04 17:49:40 +02001000 """
1001 assert_python_ok('-c', code)
1002
Victor Stinnerd554cdf2011-07-04 17:49:40 +02001003 @unittest.skipUnless(hasattr(signal, 'pthread_kill'),
1004 'need signal.pthread_kill()')
1005 def test_pthread_kill_main_thread(self):
1006 # Test that a signal can be sent to the main thread with pthread_kill()
1007 # before any other thread has been created (see issue #12392).
1008 code = """if True:
1009 import threading
1010 import signal
1011 import sys
1012
1013 def handler(signum, frame):
1014 sys.exit(3)
1015
1016 signal.signal(signal.SIGUSR1, handler)
1017 signal.pthread_kill(threading.get_ident(), signal.SIGUSR1)
1018 sys.exit(2)
1019 """
1020
1021 with spawn_python('-c', code) as process:
1022 stdout, stderr = process.communicate()
1023 exitcode = process.wait()
1024 if exitcode != 3:
1025 raise Exception("Child error (exit code %s): %s" %
1026 (exitcode, stdout))
Victor Stinnera9293352011-04-30 15:21:58 +02001027
1028
Antoine Pitrouc08177a2017-06-28 23:29:29 +02001029class StressTest(unittest.TestCase):
1030 """
1031 Stress signal delivery, especially when a signal arrives in
1032 the middle of recomputing the signal state or executing
1033 previously tripped signal handlers.
1034 """
1035
Antoine Pitrouf7d090c2017-06-29 16:40:14 +02001036 def setsig(self, signum, handler):
1037 old_handler = signal.signal(signum, handler)
1038 self.addCleanup(signal.signal, signum, old_handler)
1039
1040 def measure_itimer_resolution(self):
1041 N = 20
1042 times = []
1043
1044 def handler(signum=None, frame=None):
1045 if len(times) < N:
1046 times.append(time.perf_counter())
1047 # 1 µs is the smallest possible timer interval,
1048 # we want to measure what the concrete duration
1049 # will be on this platform
1050 signal.setitimer(signal.ITIMER_REAL, 1e-6)
1051
1052 self.addCleanup(signal.setitimer, signal.ITIMER_REAL, 0)
1053 self.setsig(signal.SIGALRM, handler)
1054 handler()
1055 while len(times) < N:
1056 time.sleep(1e-3)
1057
1058 durations = [times[i+1] - times[i] for i in range(len(times) - 1)]
1059 med = statistics.median(durations)
1060 if support.verbose:
1061 print("detected median itimer() resolution: %.6f s." % (med,))
1062 return med
1063
1064 def decide_itimer_count(self):
1065 # Some systems have poor setitimer() resolution (for example
1066 # measured around 20 ms. on FreeBSD 9), so decide on a reasonable
1067 # number of sequential timers based on that.
1068 reso = self.measure_itimer_resolution()
1069 if reso <= 1e-4:
1070 return 10000
1071 elif reso <= 1e-2:
1072 return 100
1073 else:
1074 self.skipTest("detected itimer resolution (%.3f s.) too high "
1075 "(> 10 ms.) on this platform (or system too busy)"
1076 % (reso,))
1077
Antoine Pitrouc08177a2017-06-28 23:29:29 +02001078 @unittest.skipUnless(hasattr(signal, "setitimer"),
1079 "test needs setitimer()")
1080 def test_stress_delivery_dependent(self):
1081 """
1082 This test uses dependent signal handlers.
1083 """
Antoine Pitrouf7d090c2017-06-29 16:40:14 +02001084 N = self.decide_itimer_count()
Antoine Pitrouc08177a2017-06-28 23:29:29 +02001085 sigs = []
1086
1087 def first_handler(signum, frame):
1088 # 1e-6 is the minimum non-zero value for `setitimer()`.
1089 # Choose a random delay so as to improve chances of
1090 # triggering a race condition. Ideally the signal is received
1091 # when inside critical signal-handling routines such as
1092 # Py_MakePendingCalls().
1093 signal.setitimer(signal.ITIMER_REAL, 1e-6 + random.random() * 1e-5)
1094
1095 def second_handler(signum=None, frame=None):
1096 sigs.append(signum)
1097
Antoine Pitrouc08177a2017-06-28 23:29:29 +02001098 # Here on Linux, SIGPROF > SIGALRM > SIGUSR1. By using both
1099 # ascending and descending sequences (SIGUSR1 then SIGALRM,
1100 # SIGPROF then SIGALRM), we maximize chances of hitting a bug.
Antoine Pitrouf7d090c2017-06-29 16:40:14 +02001101 self.setsig(signal.SIGPROF, first_handler)
1102 self.setsig(signal.SIGUSR1, first_handler)
1103 self.setsig(signal.SIGALRM, second_handler) # for ITIMER_REAL
Antoine Pitrouc08177a2017-06-28 23:29:29 +02001104
1105 expected_sigs = 0
1106 deadline = time.time() + 15.0
1107
1108 while expected_sigs < N:
1109 os.kill(os.getpid(), signal.SIGPROF)
1110 expected_sigs += 1
1111 # Wait for handlers to run to avoid signal coalescing
1112 while len(sigs) < expected_sigs and time.time() < deadline:
1113 time.sleep(1e-5)
1114
1115 os.kill(os.getpid(), signal.SIGUSR1)
1116 expected_sigs += 1
1117 while len(sigs) < expected_sigs and time.time() < deadline:
1118 time.sleep(1e-5)
1119
1120 # All ITIMER_REAL signals should have been delivered to the
1121 # Python handler
1122 self.assertEqual(len(sigs), N, "Some signals were lost")
1123
1124 @unittest.skipUnless(hasattr(signal, "setitimer"),
1125 "test needs setitimer()")
1126 def test_stress_delivery_simultaneous(self):
1127 """
1128 This test uses simultaneous signal handlers.
1129 """
Antoine Pitrouf7d090c2017-06-29 16:40:14 +02001130 N = self.decide_itimer_count()
Antoine Pitrouc08177a2017-06-28 23:29:29 +02001131 sigs = []
1132
1133 def handler(signum, frame):
1134 sigs.append(signum)
1135
Antoine Pitrouf7d090c2017-06-29 16:40:14 +02001136 self.setsig(signal.SIGUSR1, handler)
1137 self.setsig(signal.SIGALRM, handler) # for ITIMER_REAL
Antoine Pitrouc08177a2017-06-28 23:29:29 +02001138
1139 expected_sigs = 0
1140 deadline = time.time() + 15.0
1141
1142 while expected_sigs < N:
1143 # Hopefully the SIGALRM will be received somewhere during
1144 # initial processing of SIGUSR1.
1145 signal.setitimer(signal.ITIMER_REAL, 1e-6 + random.random() * 1e-5)
1146 os.kill(os.getpid(), signal.SIGUSR1)
1147
1148 expected_sigs += 2
1149 # Wait for handlers to run to avoid signal coalescing
1150 while len(sigs) < expected_sigs and time.time() < deadline:
1151 time.sleep(1e-5)
1152
1153 # All ITIMER_REAL signals should have been delivered to the
1154 # Python handler
1155 self.assertEqual(len(sigs), N, "Some signals were lost")
1156
1157
Zachary Ware38c707e2015-04-13 15:00:43 -05001158def tearDownModule():
1159 support.reap_children()
Thomas Woutersed03b412007-08-28 21:37:11 +00001160
1161if __name__ == "__main__":
Zachary Ware38c707e2015-04-13 15:00:43 -05001162 unittest.main()