blob: 4ae8a833b990d78f4acb5f18df440e42b6a6f596 [file] [log] [blame]
Christian Heimesdd15f6c2008-03-16 00:07:10 +00001import os
2import unittest
Guido van Rossumb26a1b41998-05-20 17:05:52 +00003import random
Benjamin Petersonee8712c2008-05-20 21:35:26 +00004from test import support
Hai Shie80697d2020-05-28 06:10:27 +08005from test.support import threading_helper
Antoine Pitrou88c60c92017-09-18 23:50:44 +02006import _thread as thread
Guido van Rossumcc544171994-04-14 20:28:41 +00007import time
Antoine Pitrou65c9c642009-10-30 17:25:12 +00008import weakref
Guido van Rossumcc544171994-04-14 20:28:41 +00009
Antoine Pitrou557934f2009-11-06 22:41:14 +000010from test import lock_tests
Guido van Rossumcc544171994-04-14 20:28:41 +000011
Christian Heimesdd15f6c2008-03-16 00:07:10 +000012NUMTASKS = 10
13NUMTRIPS = 3
Victor Stinnera15d1552017-09-12 10:49:22 -070014POLL_SLEEP = 0.010 # seconds = 10 ms
Guido van Rossumd3b68421994-05-23 12:17:36 +000015
Christian Heimesb186d002008-03-18 15:15:01 +000016_print_mutex = thread.allocate_lock()
17
Christian Heimesdd15f6c2008-03-16 00:07:10 +000018def verbose_print(arg):
19 """Helper function for printing out debugging output."""
Benjamin Petersonee8712c2008-05-20 21:35:26 +000020 if support.verbose:
Christian Heimesb186d002008-03-18 15:15:01 +000021 with _print_mutex:
22 print(arg)
Guido van Rossumcc544171994-04-14 20:28:41 +000023
Victor Stinner79ef7f82017-05-15 17:55:32 +020024
Christian Heimesdd15f6c2008-03-16 00:07:10 +000025class BasicThreadTest(unittest.TestCase):
Guido van Rossumcc544171994-04-14 20:28:41 +000026
Christian Heimesdd15f6c2008-03-16 00:07:10 +000027 def setUp(self):
28 self.done_mutex = thread.allocate_lock()
29 self.done_mutex.acquire()
30 self.running_mutex = thread.allocate_lock()
31 self.random_mutex = thread.allocate_lock()
Antoine Pitrou97115d12009-10-23 18:34:17 +000032 self.created = 0
Christian Heimesdd15f6c2008-03-16 00:07:10 +000033 self.running = 0
34 self.next_ident = 0
Guido van Rossumcc544171994-04-14 20:28:41 +000035
Hai Shie80697d2020-05-28 06:10:27 +080036 key = threading_helper.threading_setup()
37 self.addCleanup(threading_helper.threading_cleanup, *key)
Victor Stinner79ef7f82017-05-15 17:55:32 +020038
Guido van Rossumd3b68421994-05-23 12:17:36 +000039
Christian Heimesdd15f6c2008-03-16 00:07:10 +000040class ThreadRunningTests(BasicThreadTest):
41
42 def newtask(self):
43 with self.running_mutex:
44 self.next_ident += 1
45 verbose_print("creating task %s" % self.next_ident)
46 thread.start_new_thread(self.task, (self.next_ident,))
Antoine Pitrou97115d12009-10-23 18:34:17 +000047 self.created += 1
Christian Heimesdd15f6c2008-03-16 00:07:10 +000048 self.running += 1
49
50 def task(self, ident):
51 with self.random_mutex:
Christian Heimesb186d002008-03-18 15:15:01 +000052 delay = random.random() / 10000.0
53 verbose_print("task %s will run for %sus" % (ident, round(delay*1e6)))
Christian Heimesdd15f6c2008-03-16 00:07:10 +000054 time.sleep(delay)
55 verbose_print("task %s done" % ident)
56 with self.running_mutex:
57 self.running -= 1
Antoine Pitrou97115d12009-10-23 18:34:17 +000058 if self.created == NUMTASKS and self.running == 0:
Christian Heimesdd15f6c2008-03-16 00:07:10 +000059 self.done_mutex.release()
60
61 def test_starting_threads(self):
Hai Shie80697d2020-05-28 06:10:27 +080062 with threading_helper.wait_threads_exit():
Victor Stinnerff40ecd2017-09-14 13:07:24 -070063 # Basic test for thread creation.
64 for i in range(NUMTASKS):
65 self.newtask()
66 verbose_print("waiting for tasks to complete...")
67 self.done_mutex.acquire()
68 verbose_print("all tasks done")
Christian Heimesdd15f6c2008-03-16 00:07:10 +000069
70 def test_stack_size(self):
71 # Various stack size tests.
Florent Xicluna6709c0c2010-03-20 00:21:04 +000072 self.assertEqual(thread.stack_size(), 0, "initial stack size is not 0")
Christian Heimesdd15f6c2008-03-16 00:07:10 +000073
74 thread.stack_size(0)
Florent Xicluna6709c0c2010-03-20 00:21:04 +000075 self.assertEqual(thread.stack_size(), 0, "stack_size not reset to default")
Christian Heimesdd15f6c2008-03-16 00:07:10 +000076
Zachary Ware101d9e72013-12-08 00:44:27 -060077 @unittest.skipIf(os.name not in ("nt", "posix"), 'test meant for nt and posix')
Zachary Ware9fe6d862013-12-08 00:20:35 -060078 def test_nt_and_posix_stack_size(self):
Christian Heimesdd15f6c2008-03-16 00:07:10 +000079 try:
80 thread.stack_size(4096)
81 except ValueError:
82 verbose_print("caught expected ValueError setting "
83 "stack_size(4096)")
84 except thread.error:
Zachary Ware9fe6d862013-12-08 00:20:35 -060085 self.skipTest("platform does not support changing thread stack "
86 "size")
Christian Heimesdd15f6c2008-03-16 00:07:10 +000087
Zachary Ware9fe6d862013-12-08 00:20:35 -060088 fail_msg = "stack_size(%d) failed - should succeed"
89 for tss in (262144, 0x100000, 0):
90 thread.stack_size(tss)
91 self.assertEqual(thread.stack_size(), tss, fail_msg % tss)
92 verbose_print("successfully set stack_size(%d)" % tss)
Christian Heimesdd15f6c2008-03-16 00:07:10 +000093
Zachary Ware9fe6d862013-12-08 00:20:35 -060094 for tss in (262144, 0x100000):
95 verbose_print("trying stack_size = (%d)" % tss)
96 self.next_ident = 0
97 self.created = 0
Hai Shie80697d2020-05-28 06:10:27 +080098 with threading_helper.wait_threads_exit():
Victor Stinnerff40ecd2017-09-14 13:07:24 -070099 for i in range(NUMTASKS):
100 self.newtask()
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000101
Victor Stinnerff40ecd2017-09-14 13:07:24 -0700102 verbose_print("waiting for all tasks to complete")
103 self.done_mutex.acquire()
104 verbose_print("all tasks done")
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000105
Zachary Ware9fe6d862013-12-08 00:20:35 -0600106 thread.stack_size(0)
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000107
Antoine Pitrou65c9c642009-10-30 17:25:12 +0000108 def test__count(self):
109 # Test the _count() function.
110 orig = thread._count()
111 mut = thread.allocate_lock()
112 mut.acquire()
113 started = []
Victor Stinnerff40ecd2017-09-14 13:07:24 -0700114
Antoine Pitrou65c9c642009-10-30 17:25:12 +0000115 def task():
116 started.append(None)
117 mut.acquire()
118 mut.release()
Victor Stinnerff40ecd2017-09-14 13:07:24 -0700119
Hai Shie80697d2020-05-28 06:10:27 +0800120 with threading_helper.wait_threads_exit():
Victor Stinnerff40ecd2017-09-14 13:07:24 -0700121 thread.start_new_thread(task, ())
122 while not started:
123 time.sleep(POLL_SLEEP)
124 self.assertEqual(thread._count(), orig + 1)
125 # Allow the task to finish.
126 mut.release()
127 # The only reliable way to be sure that the thread ended from the
128 # interpreter's point of view is to wait for the function object to be
129 # destroyed.
130 done = []
131 wr = weakref.ref(task, lambda _: done.append(None))
132 del task
133 while not done:
134 time.sleep(POLL_SLEEP)
Serhiy Storchaka462c1f02021-09-08 18:08:57 +0300135 support.gc_collect() # For PyPy or other GCs.
Victor Stinnerff40ecd2017-09-14 13:07:24 -0700136 self.assertEqual(thread._count(), orig)
Antoine Pitrou65c9c642009-10-30 17:25:12 +0000137
Victor Stinner8b095002019-05-29 02:57:56 +0200138 def test_unraisable_exception(self):
139 def task():
140 started.release()
141 raise ValueError("task failed")
142
143 started = thread.allocate_lock()
144 with support.catch_unraisable_exception() as cm:
Hai Shie80697d2020-05-28 06:10:27 +0800145 with threading_helper.wait_threads_exit():
Victor Stinner8b095002019-05-29 02:57:56 +0200146 started.acquire()
147 thread.start_new_thread(task, ())
148 started.acquire()
149
150 self.assertEqual(str(cm.unraisable.exc_value), "task failed")
151 self.assertIs(cm.unraisable.object, task)
152 self.assertEqual(cm.unraisable.err_msg,
153 "Exception ignored in thread started by")
154 self.assertIsNotNone(cm.unraisable.exc_traceback)
155
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000156
157class Barrier:
158 def __init__(self, num_threads):
159 self.num_threads = num_threads
Fred Drake004d5e62000-10-23 17:22:08 +0000160 self.waiting = 0
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000161 self.checkin_mutex = thread.allocate_lock()
162 self.checkout_mutex = thread.allocate_lock()
163 self.checkout_mutex.acquire()
Guido van Rossumd3b68421994-05-23 12:17:36 +0000164
Fred Drake004d5e62000-10-23 17:22:08 +0000165 def enter(self):
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000166 self.checkin_mutex.acquire()
Fred Drake004d5e62000-10-23 17:22:08 +0000167 self.waiting = self.waiting + 1
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000168 if self.waiting == self.num_threads:
169 self.waiting = self.num_threads - 1
170 self.checkout_mutex.release()
Fred Drake004d5e62000-10-23 17:22:08 +0000171 return
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000172 self.checkin_mutex.release()
Guido van Rossumd3b68421994-05-23 12:17:36 +0000173
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000174 self.checkout_mutex.acquire()
Fred Drake004d5e62000-10-23 17:22:08 +0000175 self.waiting = self.waiting - 1
176 if self.waiting == 0:
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000177 self.checkin_mutex.release()
Fred Drake004d5e62000-10-23 17:22:08 +0000178 return
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000179 self.checkout_mutex.release()
Guido van Rossumd3b68421994-05-23 12:17:36 +0000180
Guido van Rossumd3b68421994-05-23 12:17:36 +0000181
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000182class BarrierTest(BasicThreadTest):
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000183
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000184 def test_barrier(self):
Hai Shie80697d2020-05-28 06:10:27 +0800185 with threading_helper.wait_threads_exit():
Victor Stinnerff40ecd2017-09-14 13:07:24 -0700186 self.bar = Barrier(NUMTASKS)
187 self.running = NUMTASKS
188 for i in range(NUMTASKS):
189 thread.start_new_thread(self.task2, (i,))
190 verbose_print("waiting for tasks to end")
191 self.done_mutex.acquire()
192 verbose_print("tasks done")
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000193
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000194 def task2(self, ident):
195 for i in range(NUMTRIPS):
196 if ident == 0:
197 # give it a good chance to enter the next
198 # barrier before the others are all out
199 # of the current one
Christian Heimesb186d002008-03-18 15:15:01 +0000200 delay = 0
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000201 else:
202 with self.random_mutex:
Christian Heimesb186d002008-03-18 15:15:01 +0000203 delay = random.random() / 10000.0
204 verbose_print("task %s will run for %sus" %
205 (ident, round(delay * 1e6)))
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000206 time.sleep(delay)
207 verbose_print("task %s entering %s" % (ident, i))
208 self.bar.enter()
209 verbose_print("task %s leaving barrier" % ident)
210 with self.running_mutex:
211 self.running -= 1
212 # Must release mutex before releasing done, else the main thread can
213 # exit and set mutex to None as part of global teardown; then
214 # mutex.release() raises AttributeError.
215 finished = self.running == 0
216 if finished:
217 self.done_mutex.release()
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000218
Antoine Pitrou557934f2009-11-06 22:41:14 +0000219class LockTests(lock_tests.LockTests):
220 locktype = thread.allocate_lock
221
222
Gregory P. Smith24cec9f2010-03-01 06:18:41 +0000223class TestForkInThread(unittest.TestCase):
224 def setUp(self):
225 self.read_fd, self.write_fd = os.pipe()
226
Victor Stinnera15d1552017-09-12 10:49:22 -0700227 @unittest.skipUnless(hasattr(os, 'fork'), 'need os.fork')
Hai Shie80697d2020-05-28 06:10:27 +0800228 @threading_helper.reap_threads
Gregory P. Smith24cec9f2010-03-01 06:18:41 +0000229 def test_forkinthread(self):
Victor Stinnera9f96872020-03-31 21:49:44 +0200230 pid = None
Gregory P. Smith24cec9f2010-03-01 06:18:41 +0000231
Victor Stinnera9f96872020-03-31 21:49:44 +0200232 def fork_thread(read_fd, write_fd):
233 nonlocal pid
Victor Stinnera15d1552017-09-12 10:49:22 -0700234
235 # fork in a thread
236 pid = os.fork()
Victor Stinnera9f96872020-03-31 21:49:44 +0200237 if pid:
238 # parent process
239 return
240
241 # child process
242 try:
243 os.close(read_fd)
244 os.write(write_fd, b"OK")
245 finally:
246 os._exit(0)
Gregory P. Smith24cec9f2010-03-01 06:18:41 +0000247
Hai Shie80697d2020-05-28 06:10:27 +0800248 with threading_helper.wait_threads_exit():
Victor Stinnera9f96872020-03-31 21:49:44 +0200249 thread.start_new_thread(fork_thread, (self.read_fd, self.write_fd))
250 self.assertEqual(os.read(self.read_fd, 2), b"OK")
251 os.close(self.write_fd)
252
253 self.assertIsNotNone(pid)
254 support.wait_process(pid, exitcode=0)
Gregory P. Smith24cec9f2010-03-01 06:18:41 +0000255
256 def tearDown(self):
257 try:
258 os.close(self.read_fd)
259 except OSError:
260 pass
261
262 try:
263 os.close(self.write_fd)
264 except OSError:
265 pass
266
267
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000268if __name__ == "__main__":
Zachary Ware38c707e2015-04-13 15:00:43 -0500269 unittest.main()