1
0
mirror of https://github.com/trezor/trezor-firmware.git synced 2024-12-16 11:28:14 +00:00

simplify loop.py, add loop.Future, supoort await

This commit is contained in:
Jan Pochyla 2016-08-05 12:32:37 +02:00 committed by Pavol Rusnak
parent 3a69524c52
commit c8c34c875b
No known key found for this signature in database
GPG Key ID: 91F3B339B9A02A3D
2 changed files with 143 additions and 120 deletions

View File

@ -6,62 +6,119 @@ from . import msg
from . import log from . import log
if __debug__: if __debug__:
# For performance stats # for performance stats
import array import array
log_delay_pos = 0 log_delay_pos = 0
log_delay_rb_len = const(10) log_delay_rb_len = const(10)
log_delay_rb = array.array('i', [0] * log_delay_rb_len) log_delay_rb = array.array('i', [0] * log_delay_rb_len)
# Touch interface paused_tasks = {} # {message interface: [task]}
TOUCH = const(256) # 0-255 is reserved for USB interfaces schedule_counter = 0
TOUCH_START = const(1) scheduled_tasks = [] # heap: [(time, counter, task, value)]
TOUCH_MOVE = const(2) MAX_SELECT_DELAY = const(1000000)
TOUCH_END = const(4)
msg_handlers = {} # Message interface -> [generator] # message interfaces:
time_queue = [] # 0-255 - USB HID
time_ticket = 0 # 256 - touch event interface
TOUCH = const(256) # interface
TOUCH_START = const(1) # event
TOUCH_MOVE = const(2) # event
TOUCH_END = const(4) # event
def schedule(gen, data=None, time=None): def schedule_task(task, value=None, time=None):
global time_ticket global schedule_counter
if not time: if time is None:
time = utime.ticks_us() time = utime.ticks_us()
heappush(time_queue, (time, time_ticket, gen, data)) heappush(scheduled_tasks, (time, schedule_counter, task, value))
time_ticket += 1 schedule_counter += 1
return gen
def unschedule(gen): def unschedule_task(task):
global time_queue global scheduled_tasks
time_queue = [entry for entry in time_queue if entry[1] is not gen] scheduled_tasks = [t for t in scheduled_tasks if t[1] is not task]
heapify(time_queue) heapify(scheduled_tasks)
def block(gen, iface): def pause_task(task, iface):
if iface in msg_handlers: paused_tasks.setdefault(iface, []).append(task)
msg_handlers[iface].append(gen)
def unpause_task(task):
for iface in paused_tasks:
if task in paused_tasks[iface]:
paused_tasks[iface].remove(task)
def run_task(task, value):
try:
if isinstance(value, Exception):
result = task.throw(value)
else: else:
msg_handlers[iface] = [gen] result = task.send(value)
except StopIteration as e:
log.debug(__name__, '%s finished', task)
except Exception as e:
log.exception(__name__, e)
else:
if isinstance(result, Syscall):
result.handle(task)
elif result is None:
schedule_task(task)
else:
log.error(__name__, '%s is unknown syscall', result)
def unblock(gen): def handle_message(message):
for iface in msg_handlers: if not paused_tasks:
if gen in msg_handlers[iface]: return
msg_handlers[iface].remove(gen) iface, *value = message
tasks = paused_tasks.pop(iface, ())
for task in tasks:
run_task(task, value)
def handle_timeout():
if not scheduled_tasks:
return
_, _, task, value = heappop(scheduled_tasks)
run_task(task, value)
def run_forever():
if __debug__:
global log_delay_pos
while True:
if scheduled_tasks:
t, _, _, _ = scheduled_tasks[0]
delay = t - utime.ticks_us()
else:
delay = MAX_SELECT_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
message = msg.select(delay)
if message:
handle_message(message)
else:
handle_timeout()
class Syscall(): class Syscall():
pass
def __iter__(self):
return (yield self)
class Sleep(Syscall): class Sleep(Syscall):
def __init__(self, us): def __init__(self, delay_us):
self.time = utime.ticks_us() + us self.time = delay_us + utime.ticks_us()
def register(self, gen): def handle(self, task):
schedule(gen, self, self.time) schedule_task(task, self, self.time)
class Select(Syscall): class Select(Syscall):
@ -69,105 +126,71 @@ class Select(Syscall):
def __init__(self, iface): def __init__(self, iface):
self.iface = iface self.iface = iface
def register(self, gen): def handle(self, task):
block(gen, self.iface) pause_task(task, self.iface)
NO_VALUE = ()
class Future(Syscall):
def __init__(self):
self.value = NO_VALUE
self.task = None
def handle(self, task):
self.task = task
if self.value is not NO_VALUE:
self._deliver()
def resolve(self, value):
if self.value is NO_VALUE:
self.value = value
if self.task is not None:
self._deliver()
def _deliver(self):
schedule_task(self.task, self.value)
class Wait(Syscall): class Wait(Syscall):
def __init__(self, gens, wait_for=1, exit_others=True): def __init__(self, children, wait_for=1, exit_others=True):
self.gens = gens self.children = children
self.wait_for = wait_for self.wait_for = wait_for
self.exit_others = exit_others self.exit_others = exit_others
self.scheduled = [] # In uPython, set() cannot contain generators self.scheduled = []
self.finished = [] self.finished = []
self.callback = None self.callback = None
def register(self, gen): def handle(self, task):
self.scheduled = [schedule(self._wait(g)) for g in self.gens] self.callback = task
self.callback = gen self.scheduled = [self._wait(c) for c in self.children]
for ct in self.scheduled:
schedule_task(ct)
def exit(self): def exit(self):
for gen in self.scheduled: for task in self.scheduled:
if gen not in self.finished and isinstance(gen, type_gen): if task not in self.finished:
unschedule(gen) unschedule_task(task)
unblock(gen) unpause_task(task)
gen.close() task.close()
def _wait(self, gen): def _wait(self, child):
try: try:
if isinstance(gen, type_gen): if isinstance(child, type_gen):
result = yield from gen result = yield from child
else: else:
result = yield gen result = yield child
except Exception as exc: except Exception as e:
self._finish(gen, exc) self._finish(child, e)
else: else:
self._finish(gen, result) self._finish(child, result)
def _finish(self, gen, result): def _finish(self, child, result):
self.finished.append(gen) self.finished.append(child)
if self.wait_for == len(self.finished) or isinstance(result, Exception): if self.wait_for == len(self.finished) or isinstance(result, Exception):
if self.exit_others: if self.exit_others:
self.exit() self.exit()
schedule(self.callback, result) schedule_task(self.callback, result)
self.callback = None
def step_task(gen, data):
if isinstance(data, Exception):
result = gen.throw(data)
else:
result = gen.send(data)
if isinstance(result, Syscall):
result.register(gen) # Execute the syscall
elif result is None:
schedule(gen) # Just call us asap
else:
raise Exception('Unhandled result %s by %s' % (result, gen))
def run_forever():
if __debug__:
global log_delay_pos, log_delay_rb, log_delay_rb_len
DELAY_MAX = const(1000000)
while True:
# Peek at how long we can sleep while waiting for an event
if time_queue:
t, _, _, _ = time_queue[0]
delay = t - utime.ticks_us()
else:
delay = DELAY_MAX
if __debug__:
# Adding 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
m = msg.select(delay)
if m:
# Run interrupt handlers right away, they have priority
iface, *data = m
tasks = msg_handlers.pop(iface, None)
if not tasks:
log.info(__name__, 'No handler for message: %s', iface)
continue
else:
# Run something from the time queue
if time_queue:
_, _, gen, data = heappop(time_queue)
tasks = (gen,)
else:
continue
# Run the tasks
for gen in tasks:
try:
step_task(gen, data)
except StopIteration as e:
log.debug(__name__, '%s finished', gen)
except Exception as e:
log.exception(__name__, e)

View File

@ -12,7 +12,7 @@ log.level = log.INFO
def perf_info_debug(): def perf_info_debug():
while True: while True:
queue = [str(x[2]).split("'")[1] for x in loop.time_queue] queue = [str(x[2]).split("'")[1] for x in loop.scheduled_tasks]
delay_avg = sum(loop.log_delay_rb) / loop.log_delay_rb_len delay_avg = sum(loop.log_delay_rb) / loop.log_delay_rb_len
delay_last = loop.log_delay_rb[loop.log_delay_pos] delay_last = loop.log_delay_rb[loop.log_delay_pos]
@ -29,13 +29,13 @@ def perf_info():
while True: while True:
gc.collect() gc.collect()
log.info(__name__, "mem_alloc: %d", gc.mem_alloc()) log.info(__name__, "mem_alloc: %d", gc.mem_alloc())
yield loop.sleep(1000000) yield loop.Sleep(1000000)
def run(main_layout): def run(main_layout):
if __debug__: if __debug__:
loop.schedule(perf_info_debug()) loop.schedule_task(perf_info_debug())
else: else:
loop.schedule(perf_info()) loop.schedule_task(perf_info())
loop.schedule(layout.set_main(main_layout)) loop.schedule_task(layout.set_main(main_layout))
loop.run_forever() loop.run_forever()