Verison 1.0.3 Release
Based on Buildbotics 0.4.14
This commit is contained in:
84
src/py/bbctrl/CommandQueue.py
Normal file
84
src/py/bbctrl/CommandQueue.py
Normal file
@@ -0,0 +1,84 @@
|
||||
################################################################################
|
||||
# #
|
||||
# This file is part of the Buildbotics firmware. #
|
||||
# #
|
||||
# Copyright (c) 2015 - 2018, Buildbotics LLC #
|
||||
# All rights reserved. #
|
||||
# #
|
||||
# This file ("the software") is free software: you can redistribute it #
|
||||
# and/or modify it under the terms of the GNU General Public License, #
|
||||
# version 2 as published by the Free Software Foundation. You should #
|
||||
# have received a copy of the GNU General Public License, version 2 #
|
||||
# along with the software. If not, see <http://www.gnu.org/licenses/>. #
|
||||
# #
|
||||
# The software is distributed in the hope that it will be useful, but #
|
||||
# WITHOUT ANY WARRANTY; without even the implied warranty of #
|
||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU #
|
||||
# Lesser General Public License for more details. #
|
||||
# #
|
||||
# You should have received a copy of the GNU Lesser General Public #
|
||||
# License along with the software. If not, see #
|
||||
# <http://www.gnu.org/licenses/>. #
|
||||
# #
|
||||
# For information regarding this software email: #
|
||||
# "Joseph Coffland" <joseph@buildbotics.com> #
|
||||
# #
|
||||
################################################################################
|
||||
|
||||
import bbctrl
|
||||
from collections import deque
|
||||
|
||||
|
||||
# 16-bit less with wrap around
|
||||
def id_less(a, b): return (1 << 15) < (a - b) & ((1 << 16) - 1)
|
||||
|
||||
|
||||
class CommandQueue():
|
||||
def __init__(self, ctrl):
|
||||
self.log = ctrl.log.get('CmdQ')
|
||||
self.log.set_level(bbctrl.log.WARNING)
|
||||
|
||||
self.lastEnqueueID = 0
|
||||
self.releaseID = 0
|
||||
self.q = deque()
|
||||
|
||||
|
||||
def is_active(self): return len(self.q)
|
||||
|
||||
|
||||
def clear(self):
|
||||
self.lastEnqueueID = 0
|
||||
self.releaseID = 0
|
||||
self.q.clear()
|
||||
|
||||
|
||||
def enqueue(self, id, cb, *args, **kwargs):
|
||||
self.log.info('add(#%d) releaseID=%d', id, self.releaseID)
|
||||
self.lastEnqueueID = id
|
||||
self.q.append([id, cb, args, kwargs])
|
||||
self._release()
|
||||
|
||||
|
||||
def _release(self):
|
||||
while len(self.q):
|
||||
id, cb, args, kwargs = self.q[0]
|
||||
|
||||
# Execute commands <= releaseID
|
||||
if id_less(self.releaseID, id): return
|
||||
|
||||
self.log.info('releasing id=%d' % id)
|
||||
self.q.popleft()
|
||||
|
||||
try:
|
||||
if cb is not None: cb(*args, **kwargs)
|
||||
except Exception:
|
||||
self.log.exception('During command queue callback')
|
||||
|
||||
|
||||
|
||||
def release(self, id):
|
||||
if id and not id_less(self.releaseID, id):
|
||||
self.log.debug('id out of order %d <= %d' % (id, self.releaseID))
|
||||
self.releaseID = id
|
||||
|
||||
self._release()
|
||||
Reference in New Issue
Block a user