1
0
mirror of https://github.com/trezor/trezor-firmware.git synced 2024-12-02 20:48:30 +00:00
trezor-firmware/src/trezor/loop.py

345 lines
9.5 KiB
Python
Raw Normal View History

'''
Implements an event loop with cooperative multitasking and async I/O. Tasks in
the form of python coroutines (either plain generators or `async` functions) are
stepped through until completion, and can get asynchronously blocked by
`yield`ing or `await`ing a syscall.
2017-09-16 13:00:31 +00:00
See `schedule`, `run`, and syscalls `sleep`, `select`, `signal` and `wait`.
'''
import utime
import utimeq
from micropython import const
from trezor import log
2017-08-14 09:08:47 +00:00
from trezor import io
after_step_hook = None # function, called after each task step
2017-09-16 13:00:31 +00:00
_QUEUE_SIZE = const(64) # maximum number of scheduled tasks
_queue = utimeq.utimeq(_QUEUE_SIZE)
_paused = {}
if __debug__:
# for performance stats
import array
log_delay_pos = 0
log_delay_rb_len = const(10)
log_delay_rb = array.array('i', [0] * log_delay_rb_len)
2017-09-16 13:00:31 +00:00
def schedule(task, value=None, deadline=None):
'''
Schedule task to be executed with `value` on given `deadline` (in
microseconds). Does not start the event loop itself, see `run`.
'''
2016-11-12 11:22:05 +00:00
if deadline is None:
deadline = utime.ticks_us()
2017-09-16 13:00:31 +00:00
_queue.push(deadline, task, value)
2017-09-16 13:00:31 +00:00
def pause(task, iface):
tasks = _paused.get(iface, None)
if tasks is None:
2018-01-22 12:07:12 +00:00
tasks = _paused[iface] = set()
tasks.add(task)
2016-05-16 15:10:12 +00:00
2017-10-10 13:32:43 +00:00
def close(task):
2018-01-22 12:07:12 +00:00
for iface in _paused:
_paused[iface].discard(task)
_queue.discard(task)
2017-10-10 13:32:43 +00:00
task.close()
def run():
'''
Loop forever, stepping through scheduled tasks and awaiting I/O events
2017-09-16 13:00:31 +00:00
inbetween. Use `schedule` first to add a coroutine to the task queue.
Tasks yield back to the scheduler on any I/O, usually by calling `await` on
a `Syscall`.
'''
2016-05-16 15:10:12 +00:00
if __debug__:
global log_delay_pos
2017-09-16 13:00:31 +00:00
max_delay = const(1000000) # usec delay if queue is empty
task_entry = [0, 0, 0] # deadline, task, value
msg_entry = [0, 0] # iface | flags, value
2017-10-24 11:59:09 +00:00
while _queue or _paused:
# compute the maximum amount of time we can wait for a message
2017-09-16 13:00:31 +00:00
if _queue:
delay = utime.ticks_diff(_queue.peektime(), utime.ticks_us())
else:
2017-09-16 13:00:31 +00:00
delay = max_delay
if __debug__:
# add current delay to ring buffer for performance stats
log_delay_rb[log_delay_pos] = delay
log_delay_pos = (log_delay_pos + 1) % log_delay_rb_len
2017-09-16 13:00:31 +00:00
if io.poll(_paused, msg_entry, delay):
# message received, run tasks paused on the interface
msg_tasks = _paused.pop(msg_entry[0], ())
for task in msg_tasks:
_step(task, msg_entry[1])
else:
# timeout occurred, run the first scheduled task
2017-09-16 13:00:31 +00:00
if _queue:
_queue.pop(task_entry)
_step(task_entry[1], task_entry[2])
2017-09-16 13:00:31 +00:00
def _step(task, value):
try:
if isinstance(value, Exception):
result = task.throw(value)
else:
result = task.send(value)
except StopIteration as e:
2017-08-21 11:22:35 +00:00
if __debug__:
log.debug(__name__, 'finish: %s', task)
except Exception as e:
2017-08-21 11:22:35 +00:00
if __debug__:
log.exception(__name__, e)
else:
if isinstance(result, Syscall):
result.handle(task)
elif result is None:
2017-09-16 13:00:31 +00:00
schedule(task)
else:
2017-08-21 11:22:35 +00:00
if __debug__:
log.error(__name__, 'unknown syscall: %s', result)
if after_step_hook:
after_step_hook()
class Syscall:
'''
When tasks want to perform any I/O, or do any sort of communication with the
scheduler, they do so through instances of a class derived from `Syscall`.
'''
def __iter__(self):
# support `yield from` or `await` on syscalls
return (yield self)
2017-09-16 13:00:31 +00:00
class sleep(Syscall):
'''
Pause current task and resume it after given delay. Although the delay is
given in microseconds, sub-millisecond precision is not guaranteed. Result
value is the calculated deadline.
Example:
2017-09-16 13:00:31 +00:00
>>> planned = await loop.sleep(1000 * 1000) # sleep for 1ms
>>> print('missed by %d us', utime.ticks_diff(utime.ticks_us(), planned))
'''
def __init__(self, delay_us):
self.delay_us = delay_us
def handle(self, task):
deadline = utime.ticks_add(utime.ticks_us(), self.delay_us)
2017-09-16 13:00:31 +00:00
schedule(task, deadline, deadline)
2017-09-16 13:00:31 +00:00
class select(Syscall):
'''
Pause current task, and resume only after a message on `msg_iface` is
received. Messages are received either from an USB interface, or the
touch display. Result value a tuple of message values.
Example:
2017-09-16 13:00:31 +00:00
>>> hid_report, = await loop.select(0xABCD) # await USB HID report
>>> event, x, y = await loop.select(io.TOUCH) # await touch event
'''
def __init__(self, msg_iface):
self.msg_iface = msg_iface
2016-04-29 23:20:57 +00:00
def handle(self, task):
2017-09-16 13:00:31 +00:00
pause(task, self.msg_iface)
_NO_VALUE = ()
2017-09-16 13:00:31 +00:00
class signal(Syscall):
'''
Pause current task, and let other running task to resume it later with a
result value or an exception.
Example:
2017-09-16 13:00:31 +00:00
>>> # in task #1:
>>> signal = loop.signal()
>>> result = await signal
>>> print('awaited result:', result)
>>> # in task #2:
>>> signal.send('hello from task #2')
>>> # prints in the next iteration of the event loop
'''
def __init__(self):
self.value = _NO_VALUE
self.task = None
def handle(self, task):
self.task = task
self._deliver()
def send(self, value):
self.value = value
self._deliver()
def _deliver(self):
if self.task is not None and self.value is not _NO_VALUE:
2017-09-16 13:00:31 +00:00
schedule(self.task, self.value)
self.task = None
self.value = _NO_VALUE
2016-05-16 15:10:12 +00:00
def __iter__(self):
try:
return (yield self)
2018-04-03 23:22:40 +00:00
except: # noqa: E722
self.task = None
raise
2017-09-16 13:00:31 +00:00
class wait(Syscall):
'''
Execute one or more children tasks and wait until one or more of them exit.
2017-09-16 13:00:31 +00:00
Return value of `wait` is the return value of task that triggered the
completion. By default, `wait` returns after the first child completes, and
other running children are killed (by cancelling any pending schedules and
calling `close()`).
Example:
2017-09-16 13:00:31 +00:00
>>> # async def wait_for_touch(): ...
>>> # async def animate_logo(): ...
>>> touch_task = wait_for_touch()
>>> animation_task = animate_logo()
>>> waiter = loop.wait(touch_task, animation_task)
>>> result = await waiter
>>> if animation_task in waiter.finished:
>>> print('animation task returned', result)
>>> else:
>>> print('touch task returned', result)
Note: You should not directly `yield` a `wait` instance, see logic in
`wait.__iter__` for explanation. Always use `await`.
'''
2017-09-16 13:00:31 +00:00
def __init__(self, *children, wait_for=1, exit_others=True):
self.children = children
2016-05-02 14:06:08 +00:00
self.wait_for = wait_for
self.exit_others = exit_others
self.scheduled = None # list of scheduled wrapper tasks
self.finished = None # list of children that finished
2016-05-02 14:06:08 +00:00
self.callback = None
def handle(self, task):
self.callback = task
self.finished = []
self.scheduled = [self._wait(c) for c in self.children]
for ct in self.scheduled:
2017-09-16 13:00:31 +00:00
schedule(ct)
2016-05-02 14:06:08 +00:00
def exit(self):
for ct in self.scheduled:
close(ct)
async def _wait(self, child):
try:
result = await child
except Exception as e:
self._finish(child, e)
else:
self._finish(child, result)
2016-04-29 23:20:57 +00:00
def _finish(self, child, result):
self.finished.append(child)
if self.wait_for == len(self.finished) or isinstance(result, Exception):
schedule(self.callback, result)
2016-05-02 14:06:08 +00:00
if self.exit_others:
self.exit()
def __iter__(self):
try:
return (yield self)
except: # noqa: E722
2016-10-06 12:41:50 +00:00
# exception was raised on the waiting task externally with
# close() or throw(), kill the children tasks and re-raise
self.exit()
raise
2017-06-12 16:16:06 +00:00
2017-09-16 13:00:31 +00:00
class put(Syscall):
2017-07-04 16:09:08 +00:00
2017-09-16 13:00:31 +00:00
def __init__(self, ch, value=None):
self.ch = ch
2017-07-04 16:09:08 +00:00
self.value = value
def __call__(self, value):
self.value = value
return self
def handle(self, task):
2017-09-16 13:00:31 +00:00
self.ch.schedule_put(schedule, task, self.value)
2017-07-04 16:09:08 +00:00
2017-09-16 13:00:31 +00:00
class take(Syscall):
2017-07-04 16:09:08 +00:00
2017-09-16 13:00:31 +00:00
def __init__(self, ch):
self.ch = ch
2017-07-04 16:09:08 +00:00
def __call__(self):
return self
def handle(self, task):
2017-09-16 13:00:31 +00:00
if self.ch.schedule_take(schedule, task) and self.ch.id is not None:
pause(self.ch, self.ch.id)
2017-07-04 16:09:08 +00:00
2017-09-16 13:00:31 +00:00
class chan:
2017-07-04 16:09:08 +00:00
def __init__(self, id=None):
self.id = id
self.putters = []
self.takers = []
2017-09-16 13:00:31 +00:00
self.put = put(self)
self.take = take(self)
2017-07-04 16:09:08 +00:00
def schedule_publish(self, schedule, value):
if self.takers:
for taker in self.takers:
schedule(taker, value)
self.takers.clear()
return True
else:
return False
def schedule_put(self, schedule, putter, value):
if self.takers:
taker = self.takers.pop(0)
schedule(taker, value)
schedule(putter, value)
return True
else:
self.putters.append((putter, value))
return False
def schedule_take(self, schedule, taker):
if self.putters:
putter, value = self.putters.pop(0)
schedule(taker, value)
schedule(putter, value)
return True
else:
self.takers.append(taker)
return False