aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorKevin O'Connor <kevin@koconnor.net>2019-06-07 22:41:58 -0400
committerKevin O'Connor <kevin@koconnor.net>2019-06-09 12:55:56 -0400
commit0546c18d27ba47a336c9be409a8dc9d84d99ed3a (patch)
tree1922b4595c09e25f25abe2cc9696db525346acd2
parent61fc57e659f765680068bddd0d03d6ff637c6c6a (diff)
downloadkutter-0546c18d27ba47a336c9be409a8dc9d84d99ed3a.tar.gz
kutter-0546c18d27ba47a336c9be409a8dc9d84d99ed3a.tar.xz
kutter-0546c18d27ba47a336c9be409a8dc9d84d99ed3a.zip
reactor: Add support for mutexes
Add a mutex object that implements simple queing mutual exclusion for greenlets. Signed-off-by: Kevin O'Connor <kevin@koconnor.net>
-rw-r--r--klippy/reactor.py34
1 files changed, 33 insertions, 1 deletions
diff --git a/klippy/reactor.py b/klippy/reactor.py
index 3c43f40a..bbb2c27d 100644
--- a/klippy/reactor.py
+++ b/klippy/reactor.py
@@ -1,6 +1,6 @@
# File descriptor and timer event helper
#
-# Copyright (C) 2016-2018 Kevin O'Connor <kevin@koconnor.net>
+# Copyright (C) 2016-2019 Kevin O'Connor <kevin@koconnor.net>
#
# This file may be distributed under the terms of the GNU GPLv3 license.
import os, select, math, time, Queue
@@ -34,6 +34,35 @@ class ReactorGreenlet(greenlet.greenlet):
greenlet.greenlet.__init__(self, run=run)
self.timer = None
+class ReactorMutex:
+ def __init__(self, reactor, is_locked):
+ self.reactor = reactor
+ self.is_locked = is_locked
+ self.next_pending = False
+ self.queue = []
+ self.lock = self.__enter__
+ self.unlock = self.__exit__
+ def test(self):
+ return self.is_locked
+ def __enter__(self):
+ if not self.is_locked:
+ self.is_locked = True
+ return
+ g = greenlet.getcurrent()
+ self.queue.append(g)
+ while 1:
+ self.reactor.pause(self.reactor.NEVER)
+ if self.next_pending and self.queue[0] is g:
+ self.next_pending = False
+ self.queue.pop(0)
+ return
+ def __exit__(self, type=None, value=None, tb=None):
+ if not self.queue:
+ self.is_locked = False
+ return
+ self.next_pending = True
+ self.reactor.update_timer(self.queue[0].timer, self.reactor.NOW)
+
class SelectReactor:
NOW = 0.
NEVER = 9999999999999999.
@@ -146,6 +175,9 @@ class SelectReactor:
self._g_dispatch.switch(self.NEVER)
# This greenlet was reactivated - prepare for main processing loop
self._g_dispatch = g_old
+ # Mutexes
+ def mutex(self, is_locked=False):
+ return ReactorMutex(self, is_locked)
# File descriptors
def register_fd(self, fd, callback):
handler = ReactorFileHandler(fd, callback)