2013-09-03 18:01:33 -04:00
|
|
|
from migen.fhdl.std import *
|
|
|
|
from migen.flow.actor import *
|
|
|
|
from migen.genlib import fifo
|
|
|
|
|
2015-04-13 14:45:35 -04:00
|
|
|
|
2013-09-04 11:22:50 -04:00
|
|
|
class _FIFOActor(Module):
|
2015-04-13 14:07:07 -04:00
|
|
|
def __init__(self, fifo_class, layout, depth):
|
|
|
|
self.sink = Sink(layout)
|
|
|
|
self.source = Source(layout)
|
|
|
|
self.busy = Signal()
|
|
|
|
|
|
|
|
###
|
|
|
|
|
|
|
|
description = self.sink.description
|
|
|
|
fifo_layout = [
|
|
|
|
("payload", description.payload_layout),
|
|
|
|
# Note : Can be optimized by passing parameters
|
|
|
|
# in another fifo. We will only have one
|
|
|
|
# data per packet.
|
|
|
|
("param", description.param_layout)
|
|
|
|
]
|
|
|
|
if description.packetized:
|
|
|
|
fifo_layout += [("sop", 1), ("eop", 1)]
|
|
|
|
|
|
|
|
self.submodules.fifo = fifo_class(fifo_layout, depth)
|
|
|
|
|
|
|
|
self.comb += [
|
|
|
|
self.sink.ack.eq(self.fifo.writable),
|
|
|
|
self.fifo.we.eq(self.sink.stb),
|
|
|
|
self.fifo.din.payload.eq(self.sink.payload),
|
|
|
|
self.fifo.din.param.eq(self.sink.param),
|
|
|
|
|
|
|
|
self.source.stb.eq(self.fifo.readable),
|
|
|
|
self.source.payload.eq(self.fifo.dout.payload),
|
|
|
|
self.source.param.eq(self.fifo.dout.param),
|
|
|
|
self.fifo.re.eq(self.source.ack)
|
|
|
|
]
|
|
|
|
if description.packetized:
|
|
|
|
self.comb += [
|
|
|
|
self.fifo.din.sop.eq(self.sink.sop),
|
|
|
|
self.fifo.din.eop.eq(self.sink.eop),
|
|
|
|
self.source.sop.eq(self.fifo.dout.sop),
|
|
|
|
self.source.eop.eq(self.fifo.dout.eop)
|
|
|
|
]
|
2013-09-03 18:01:33 -04:00
|
|
|
|
2015-04-13 14:45:35 -04:00
|
|
|
|
2013-09-04 11:22:50 -04:00
|
|
|
class SyncFIFO(_FIFOActor):
|
2015-04-13 14:07:07 -04:00
|
|
|
def __init__(self, layout, depth, buffered=False):
|
|
|
|
_FIFOActor.__init__(
|
|
|
|
self,
|
|
|
|
fifo.SyncFIFOBuffered if buffered else fifo.SyncFIFO,
|
|
|
|
layout, depth)
|
2013-09-03 18:01:33 -04:00
|
|
|
|
2015-04-13 14:45:35 -04:00
|
|
|
|
2013-09-04 11:22:50 -04:00
|
|
|
class AsyncFIFO(_FIFOActor):
|
2015-04-13 14:07:07 -04:00
|
|
|
def __init__(self, layout, depth):
|
|
|
|
_FIFOActor.__init__(self, fifo.AsyncFIFO, layout, depth)
|
2015-07-24 07:02:54 -04:00
|
|
|
|
|
|
|
|
|
|
|
def FIFO(layout, depth, buffered=False,
|
|
|
|
sink_cd="sys", source_cd="sys"):
|
|
|
|
if sink_cd != source_cd:
|
|
|
|
if buffered:
|
|
|
|
ValueError("AsyncFIFO does not support buffered mode")
|
|
|
|
fifo = AsyncFIFO(layout, depth)
|
|
|
|
return ClockDomainsRenamer({"write": sink_cd, "read": source_cd})(fifo)
|
|
|
|
else:
|
|
|
|
fifo = SyncFIFO(layout, depth, buffered)
|
|
|
|
return ClockDomainsRenamer(sink_cd)(fifo)
|