summaryrefslogtreecommitdiff
path: root/eventlet/semaphore.py
diff options
context:
space:
mode:
Diffstat (limited to 'eventlet/semaphore.py')
-rw-r--r--eventlet/semaphore.py28
1 files changed, 22 insertions, 6 deletions
diff --git a/eventlet/semaphore.py b/eventlet/semaphore.py
index 73dbbc1..962b304 100644
--- a/eventlet/semaphore.py
+++ b/eventlet/semaphore.py
@@ -1,4 +1,7 @@
from __future__ import with_statement
+
+import collections
+
from eventlet import greenthread
from eventlet import hubs
from eventlet.timeout import Timeout
@@ -35,7 +38,7 @@ class Semaphore(object):
if value < 0:
raise ValueError("Semaphore must be initialized with a positive "
"number, got %s" % value)
- self._waiters = set()
+ self._waiters = collections.deque()
def __repr__(self):
params = (self.__class__.__name__, hex(id(self)),
@@ -80,8 +83,12 @@ class Semaphore(object):
raise ValueError("can't specify timeout for non-blocking acquire")
if not blocking and self.locked():
return False
- if self.counter <= 0:
- self._waiters.add(greenthread.getcurrent())
+
+ current_thread = greenthread.getcurrent()
+
+ if self.counter <= 0 or self._waiters:
+ if current_thread not in self._waiters:
+ self._waiters.append(current_thread)
try:
if timeout is not None:
ok = False
@@ -92,10 +99,19 @@ class Semaphore(object):
if not ok:
return False
else:
- while self.counter <= 0:
+ # If someone else is already in this wait loop, give them
+ # a chance to get out.
+ while True:
hubs.get_hub().switch()
+ if self.counter > 0:
+ break
finally:
- self._waiters.discard(greenthread.getcurrent())
+ try:
+ self._waiters.remove(current_thread)
+ except ValueError:
+ # Fine if its already been dropped.
+ pass
+
self.counter -= 1
return True
@@ -117,7 +133,7 @@ class Semaphore(object):
def _do_acquire(self):
if self._waiters and self.counter > 0:
- waiter = self._waiters.pop()
+ waiter = self._waiters.popleft()
waiter.switch()
def __exit__(self, typ, val, tb):