blob: 134ac3b7ca7c17ab02596967a00b2eaf31dbec15 [file] [log] [blame]
Brett Cannon66865d22008-03-13 20:27:00 +00001import os
2import unittest
Guido van Rossumb26a1b41998-05-20 17:05:52 +00003import random
Brett Cannon66865d22008-03-13 20:27:00 +00004from test import test_support
Guido van Rossumcc544171994-04-14 20:28:41 +00005import thread
6import time
Antoine Pitrou59c44f32009-10-30 17:07:08 +00007import weakref
Guido van Rossumcc544171994-04-14 20:28:41 +00008
Antoine Pitrouc98efe02009-11-06 22:34:35 +00009from test import lock_tests
Guido van Rossumcc544171994-04-14 20:28:41 +000010
Brett Cannon66865d22008-03-13 20:27:00 +000011NUMTASKS = 10
12NUMTRIPS = 3
Guido van Rossumd3b68421994-05-23 12:17:36 +000013
Guido van Rossumcc544171994-04-14 20:28:41 +000014
Jeffrey Yasskina1458532008-03-18 04:56:06 +000015_print_mutex = thread.allocate_lock()
16
Brett Cannon66865d22008-03-13 20:27:00 +000017def verbose_print(arg):
18 """Helper function for printing out debugging output."""
19 if test_support.verbose:
Jeffrey Yasskina1458532008-03-18 04:56:06 +000020 with _print_mutex:
21 print arg
Guido van Rossumcc544171994-04-14 20:28:41 +000022
Guido van Rossumcc544171994-04-14 20:28:41 +000023
Brett Cannon66865d22008-03-13 20:27:00 +000024class BasicThreadTest(unittest.TestCase):
Guido van Rossumd3b68421994-05-23 12:17:36 +000025
Brett Cannon66865d22008-03-13 20:27:00 +000026 def setUp(self):
27 self.done_mutex = thread.allocate_lock()
28 self.done_mutex.acquire()
29 self.running_mutex = thread.allocate_lock()
30 self.random_mutex = thread.allocate_lock()
Antoine Pitroub5cf8a02009-10-23 18:32:15 +000031 self.created = 0
Brett Cannon66865d22008-03-13 20:27:00 +000032 self.running = 0
33 self.next_ident = 0
34
35
36class ThreadRunningTests(BasicThreadTest):
37
38 def newtask(self):
39 with self.running_mutex:
40 self.next_ident += 1
41 verbose_print("creating task %s" % self.next_ident)
42 thread.start_new_thread(self.task, (self.next_ident,))
Antoine Pitroub5cf8a02009-10-23 18:32:15 +000043 self.created += 1
Brett Cannon66865d22008-03-13 20:27:00 +000044 self.running += 1
45
46 def task(self, ident):
47 with self.random_mutex:
Jeffrey Yasskina1458532008-03-18 04:56:06 +000048 delay = random.random() / 10000.0
49 verbose_print("task %s will run for %sus" % (ident, round(delay*1e6)))
Brett Cannon66865d22008-03-13 20:27:00 +000050 time.sleep(delay)
51 verbose_print("task %s done" % ident)
52 with self.running_mutex:
53 self.running -= 1
Antoine Pitroub5cf8a02009-10-23 18:32:15 +000054 if self.created == NUMTASKS and self.running == 0:
Brett Cannon66865d22008-03-13 20:27:00 +000055 self.done_mutex.release()
56
57 def test_starting_threads(self):
58 # Basic test for thread creation.
59 for i in range(NUMTASKS):
60 self.newtask()
61 verbose_print("waiting for tasks to complete...")
62 self.done_mutex.acquire()
63 verbose_print("all tasks done")
64
65 def test_stack_size(self):
66 # Various stack size tests.
67 self.assertEquals(thread.stack_size(), 0, "intial stack size is not 0")
68
69 thread.stack_size(0)
70 self.assertEquals(thread.stack_size(), 0, "stack_size not reset to default")
71
72 if os.name not in ("nt", "os2", "posix"):
73 return
74
75 tss_supported = True
76 try:
77 thread.stack_size(4096)
78 except ValueError:
79 verbose_print("caught expected ValueError setting "
80 "stack_size(4096)")
81 except thread.error:
82 tss_supported = False
83 verbose_print("platform does not support changing thread stack "
84 "size")
85
86 if tss_supported:
87 fail_msg = "stack_size(%d) failed - should succeed"
88 for tss in (262144, 0x100000, 0):
89 thread.stack_size(tss)
90 self.assertEquals(thread.stack_size(), tss, fail_msg % tss)
91 verbose_print("successfully set stack_size(%d)" % tss)
92
93 for tss in (262144, 0x100000):
94 verbose_print("trying stack_size = (%d)" % tss)
95 self.next_ident = 0
Antoine Pitroub5cf8a02009-10-23 18:32:15 +000096 self.created = 0
Brett Cannon66865d22008-03-13 20:27:00 +000097 for i in range(NUMTASKS):
98 self.newtask()
99
100 verbose_print("waiting for all tasks to complete")
101 self.done_mutex.acquire()
102 verbose_print("all tasks done")
103
104 thread.stack_size(0)
105
Antoine Pitrou59c44f32009-10-30 17:07:08 +0000106 def test__count(self):
107 # Test the _count() function.
108 orig = thread._count()
109 mut = thread.allocate_lock()
110 mut.acquire()
111 started = []
112 def task():
113 started.append(None)
114 mut.acquire()
115 mut.release()
116 thread.start_new_thread(task, ())
117 while not started:
118 time.sleep(0.01)
119 self.assertEquals(thread._count(), orig + 1)
120 # Allow the task to finish.
121 mut.release()
122 # The only reliable way to be sure that the thread ended from the
123 # interpreter's point of view is to wait for the function object to be
124 # destroyed.
125 done = []
126 wr = weakref.ref(task, lambda _: done.append(None))
127 del task
128 while not done:
129 time.sleep(0.01)
130 self.assertEquals(thread._count(), orig)
131
Brett Cannon66865d22008-03-13 20:27:00 +0000132
133class Barrier:
134 def __init__(self, num_threads):
135 self.num_threads = num_threads
Fred Drake004d5e62000-10-23 17:22:08 +0000136 self.waiting = 0
Brett Cannon66865d22008-03-13 20:27:00 +0000137 self.checkin_mutex = thread.allocate_lock()
138 self.checkout_mutex = thread.allocate_lock()
139 self.checkout_mutex.acquire()
Guido van Rossumd3b68421994-05-23 12:17:36 +0000140
Fred Drake004d5e62000-10-23 17:22:08 +0000141 def enter(self):
Brett Cannon66865d22008-03-13 20:27:00 +0000142 self.checkin_mutex.acquire()
Fred Drake004d5e62000-10-23 17:22:08 +0000143 self.waiting = self.waiting + 1
Brett Cannon66865d22008-03-13 20:27:00 +0000144 if self.waiting == self.num_threads:
145 self.waiting = self.num_threads - 1
146 self.checkout_mutex.release()
Fred Drake004d5e62000-10-23 17:22:08 +0000147 return
Brett Cannon66865d22008-03-13 20:27:00 +0000148 self.checkin_mutex.release()
Guido van Rossumd3b68421994-05-23 12:17:36 +0000149
Brett Cannon66865d22008-03-13 20:27:00 +0000150 self.checkout_mutex.acquire()
Fred Drake004d5e62000-10-23 17:22:08 +0000151 self.waiting = self.waiting - 1
152 if self.waiting == 0:
Brett Cannon66865d22008-03-13 20:27:00 +0000153 self.checkin_mutex.release()
Fred Drake004d5e62000-10-23 17:22:08 +0000154 return
Brett Cannon66865d22008-03-13 20:27:00 +0000155 self.checkout_mutex.release()
Guido van Rossumd3b68421994-05-23 12:17:36 +0000156
Guido van Rossumd3b68421994-05-23 12:17:36 +0000157
Brett Cannon66865d22008-03-13 20:27:00 +0000158class BarrierTest(BasicThreadTest):
Andrew MacIntyre92913322006-06-13 15:04:24 +0000159
Brett Cannon66865d22008-03-13 20:27:00 +0000160 def test_barrier(self):
161 self.bar = Barrier(NUMTASKS)
162 self.running = NUMTASKS
163 for i in range(NUMTASKS):
164 thread.start_new_thread(self.task2, (i,))
165 verbose_print("waiting for tasks to end")
166 self.done_mutex.acquire()
167 verbose_print("tasks done")
Andrew MacIntyre92913322006-06-13 15:04:24 +0000168
Brett Cannon66865d22008-03-13 20:27:00 +0000169 def task2(self, ident):
170 for i in range(NUMTRIPS):
171 if ident == 0:
172 # give it a good chance to enter the next
173 # barrier before the others are all out
174 # of the current one
Jeffrey Yasskina1458532008-03-18 04:56:06 +0000175 delay = 0
Brett Cannon66865d22008-03-13 20:27:00 +0000176 else:
177 with self.random_mutex:
Jeffrey Yasskina1458532008-03-18 04:56:06 +0000178 delay = random.random() / 10000.0
179 verbose_print("task %s will run for %sus" %
180 (ident, round(delay * 1e6)))
Brett Cannon66865d22008-03-13 20:27:00 +0000181 time.sleep(delay)
182 verbose_print("task %s entering %s" % (ident, i))
183 self.bar.enter()
184 verbose_print("task %s leaving barrier" % ident)
185 with self.running_mutex:
186 self.running -= 1
187 # Must release mutex before releasing done, else the main thread can
188 # exit and set mutex to None as part of global teardown; then
189 # mutex.release() raises AttributeError.
190 finished = self.running == 0
191 if finished:
192 self.done_mutex.release()
Andrew MacIntyre92913322006-06-13 15:04:24 +0000193
Andrew MacIntyre92913322006-06-13 15:04:24 +0000194
Antoine Pitrouc98efe02009-11-06 22:34:35 +0000195class LockTests(lock_tests.LockTests):
196 locktype = thread.allocate_lock
197
198
Brett Cannon66865d22008-03-13 20:27:00 +0000199def test_main():
Antoine Pitrouc98efe02009-11-06 22:34:35 +0000200 test_support.run_unittest(ThreadRunningTests, BarrierTest, LockTests)
Andrew MacIntyre92913322006-06-13 15:04:24 +0000201
Brett Cannon66865d22008-03-13 20:27:00 +0000202if __name__ == "__main__":
203 test_main()