litex/migen/genlib/fsm.py

76 lines
2 KiB
Python
Raw Normal View History

2013-06-25 16:17:39 -04:00
from collections import OrderedDict
from migen.fhdl.std import *
2013-06-25 16:17:39 -04:00
from migen.fhdl.module import FinalizeError
from migen.fhdl.visit import NodeTransformer
2012-01-09 10:28:48 -05:00
2013-06-25 16:17:39 -04:00
class AnonymousState:
pass
# do not use namedtuple here as it inherits tuple
# and the latter is used elsewhere in FHDL
class NextState:
def __init__(self, state):
self.state = state
class _LowerNextState(NodeTransformer):
def __init__(self, next_state_signal, encoding, aliases):
self.next_state_signal = next_state_signal
self.encoding = encoding
self.aliases = aliases
2012-03-17 19:09:40 -04:00
2013-06-25 16:17:39 -04:00
def visit_unknown(self, node):
if isinstance(node, NextState):
try:
actual_state = self.aliases[node.state]
except KeyError:
actual_state = node.state
return self.next_state_signal.eq(self.encoding[actual_state])
else:
return node
class FSM(Module):
2013-07-22 04:09:12 -04:00
def __init__(self, reset_state=None):
2013-06-25 16:17:39 -04:00
self.actions = OrderedDict()
self.state_aliases = dict()
2013-07-22 04:09:12 -04:00
self.reset_state = reset_state
2013-06-25 16:17:39 -04:00
2012-01-09 10:28:48 -05:00
def act(self, state, *statements):
2013-06-25 16:17:39 -04:00
if self.finalized:
raise FinalizeError
if state not in self.actions:
self.actions[state] = []
2012-11-28 17:18:43 -05:00
self.actions[state] += statements
2013-06-25 16:17:39 -04:00
def delayed_enter(self, name, target, delay):
if self.finalized:
raise FinalizeError
if delay:
state = name
for i in range(delay):
if i == delay - 1:
next_state = target
else:
next_state = AnonymousState()
self.act(state, NextState(next_state))
state = next_state
else:
self.state_aliases[name] = target
2012-01-09 10:28:48 -05:00
2013-06-25 16:17:39 -04:00
def do_finalize(self):
nstates = len(self.actions)
self.encoding = dict((s, n) for n, s in enumerate(self.actions.keys()))
self.state = Signal(max=nstates)
2013-07-22 04:09:12 -04:00
if self.reset_state is not None:
self.state.reset = self.encoding[self.reset_state]
2013-06-25 16:17:39 -04:00
self.next_state = Signal(max=nstates)
lns = _LowerNextState(self.next_state, self.encoding, self.state_aliases)
cases = dict((self.encoding[k], lns.visit(v)) for k, v in self.actions.items() if v)
self.comb += [
self.next_state.eq(self.state),
Case(self.state, cases)
2012-01-09 10:28:48 -05:00
]
2013-06-25 16:17:39 -04:00
self.sync += self.state.eq(self.next_state)