summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--Lib/test/test_threading.py18
-rw-r--r--Lib/threading.py8
2 files changed, 23 insertions, 3 deletions
diff --git a/Lib/test/test_threading.py b/Lib/test/test_threading.py
index 826acbb..4d30ee4 100644
--- a/Lib/test/test_threading.py
+++ b/Lib/test/test_threading.py
@@ -599,6 +599,24 @@ class ThreadTests(BaseTestCase):
time.sleep(0.01)
self.assertIn(LOOKING_FOR, repr(t)) # we waited at least 5 seconds
+ def test_BoundedSemaphore_limit(self):
+ # BoundedSemaphore should raise ValueError if released too often.
+ for limit in range(1, 10):
+ bs = threading.BoundedSemaphore(limit)
+ threads = [threading.Thread(target=bs.acquire)
+ for _ in range(limit)]
+ for t in threads:
+ t.start()
+ for t in threads:
+ t.join()
+ threads = [threading.Thread(target=bs.release)
+ for _ in range(limit)]
+ for t in threads:
+ t.start()
+ for t in threads:
+ t.join()
+ self.assertRaises(ValueError, bs.release)
+
class ThreadJoinOnShutdown(BaseTestCase):
def _run_and_join(self, script):
diff --git a/Lib/threading.py b/Lib/threading.py
index 26d1018..b99d0e9 100644
--- a/Lib/threading.py
+++ b/Lib/threading.py
@@ -289,9 +289,11 @@ class BoundedSemaphore(Semaphore):
self._initial_value = value
def release(self):
- if self._value >= self._initial_value:
- raise ValueError("Semaphore released too many times")
- return Semaphore.release(self)
+ with self._cond:
+ if self._value >= self._initial_value:
+ raise ValueError("Semaphore released too many times")
+ self._value += 1
+ self._cond.notify()
class Event: