lib.fifo: add r_level
and w_level
to all FIFOs
This commit is contained in:
parent
f6f9d68f24
commit
73f672f57c
|
@ -32,6 +32,8 @@ class FIFOInterface:
|
||||||
a new entry.
|
a new entry.
|
||||||
w_en : in
|
w_en : in
|
||||||
Write strobe. Latches ``w_data`` into the queue. Does nothing if ``w_rdy`` is not asserted.
|
Write strobe. Latches ``w_data`` into the queue. Does nothing if ``w_rdy`` is not asserted.
|
||||||
|
w_level : out
|
||||||
|
Number of unread entries.
|
||||||
{w_attributes}
|
{w_attributes}
|
||||||
r_data : out, width
|
r_data : out, width
|
||||||
Output data. {r_data_valid}
|
Output data. {r_data_valid}
|
||||||
|
@ -41,6 +43,8 @@ class FIFOInterface:
|
||||||
r_en : in
|
r_en : in
|
||||||
Read strobe. Makes the next entry (if any) available on ``r_data`` at the next cycle.
|
Read strobe. Makes the next entry (if any) available on ``r_data`` at the next cycle.
|
||||||
Does nothing if ``r_rdy`` is not asserted.
|
Does nothing if ``r_rdy`` is not asserted.
|
||||||
|
r_level : out
|
||||||
|
Number of unread entries.
|
||||||
{r_attributes}
|
{r_attributes}
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
@ -74,10 +78,12 @@ class FIFOInterface:
|
||||||
self.w_data = Signal(width, reset_less=True)
|
self.w_data = Signal(width, reset_less=True)
|
||||||
self.w_rdy = Signal() # writable; not full
|
self.w_rdy = Signal() # writable; not full
|
||||||
self.w_en = Signal()
|
self.w_en = Signal()
|
||||||
|
self.w_level = Signal(range(depth + 1))
|
||||||
|
|
||||||
self.r_data = Signal(width, reset_less=True)
|
self.r_data = Signal(width, reset_less=True)
|
||||||
self.r_rdy = Signal() # readable; not empty
|
self.r_rdy = Signal() # readable; not empty
|
||||||
self.r_en = Signal()
|
self.r_en = Signal()
|
||||||
|
self.r_level = Signal(range(depth + 1))
|
||||||
|
|
||||||
|
|
||||||
def _incr(signal, modulo):
|
def _incr(signal, modulo):
|
||||||
|
@ -106,10 +112,7 @@ class SyncFIFO(Elaboratable, FIFOInterface):
|
||||||
cycle after ``r_rdy`` and ``r_en`` have been asserted.
|
cycle after ``r_rdy`` and ``r_en`` have been asserted.
|
||||||
""".strip(),
|
""".strip(),
|
||||||
attributes="",
|
attributes="",
|
||||||
r_attributes="""
|
r_attributes="",
|
||||||
level : out
|
|
||||||
Number of unread entries.
|
|
||||||
""".strip(),
|
|
||||||
w_attributes="")
|
w_attributes="")
|
||||||
|
|
||||||
def __init__(self, *, width, depth, fwft=True):
|
def __init__(self, *, width, depth, fwft=True):
|
||||||
|
@ -128,7 +131,9 @@ class SyncFIFO(Elaboratable, FIFOInterface):
|
||||||
|
|
||||||
m.d.comb += [
|
m.d.comb += [
|
||||||
self.w_rdy.eq(self.level != self.depth),
|
self.w_rdy.eq(self.level != self.depth),
|
||||||
self.r_rdy.eq(self.level != 0)
|
self.r_rdy.eq(self.level != 0),
|
||||||
|
self.w_level.eq(self.level),
|
||||||
|
self.r_level.eq(self.level),
|
||||||
]
|
]
|
||||||
|
|
||||||
do_read = self.r_rdy & self.r_en
|
do_read = self.r_rdy & self.r_en
|
||||||
|
@ -144,7 +149,7 @@ class SyncFIFO(Elaboratable, FIFOInterface):
|
||||||
m.d.comb += [
|
m.d.comb += [
|
||||||
w_port.addr.eq(produce),
|
w_port.addr.eq(produce),
|
||||||
w_port.data.eq(self.w_data),
|
w_port.data.eq(self.w_data),
|
||||||
w_port.en.eq(self.w_en & self.w_rdy)
|
w_port.en.eq(self.w_en & self.w_rdy),
|
||||||
]
|
]
|
||||||
with m.If(do_write):
|
with m.If(do_write):
|
||||||
m.d.sync += produce.eq(_incr(produce, self.depth))
|
m.d.sync += produce.eq(_incr(produce, self.depth))
|
||||||
|
@ -248,7 +253,11 @@ class SyncFIFOBuffered(Elaboratable, FIFOInterface):
|
||||||
with m.Elif(self.r_en):
|
with m.Elif(self.r_en):
|
||||||
m.d.sync += self.r_rdy.eq(0)
|
m.d.sync += self.r_rdy.eq(0)
|
||||||
|
|
||||||
m.d.comb += self.level.eq(fifo.level + self.r_rdy)
|
m.d.comb += [
|
||||||
|
self.level.eq(fifo.level + self.r_rdy),
|
||||||
|
self.w_level.eq(self.level),
|
||||||
|
self.r_level.eq(self.level),
|
||||||
|
]
|
||||||
|
|
||||||
return m
|
return m
|
||||||
|
|
||||||
|
@ -350,6 +359,18 @@ class AsyncFIFO(Elaboratable, FIFOInterface):
|
||||||
m.d.comb += consume_enc.i.eq(consume_r_nxt)
|
m.d.comb += consume_enc.i.eq(consume_r_nxt)
|
||||||
m.d[self._r_domain] += consume_r_gry.eq(consume_enc.o)
|
m.d[self._r_domain] += consume_r_gry.eq(consume_enc.o)
|
||||||
|
|
||||||
|
consume_w_bin = Signal(self._ctr_bits)
|
||||||
|
consume_dec = m.submodules.consume_dec = \
|
||||||
|
GrayDecoder(self._ctr_bits)
|
||||||
|
m.d.comb += consume_dec.i.eq(consume_w_gry),
|
||||||
|
m.d[self._w_domain] += consume_w_bin.eq(consume_dec.o)
|
||||||
|
|
||||||
|
produce_r_bin = Signal(self._ctr_bits)
|
||||||
|
produce_dec = m.submodules.produce_dec = \
|
||||||
|
GrayDecoder(self._ctr_bits)
|
||||||
|
m.d.comb += produce_dec.i.eq(produce_r_gry),
|
||||||
|
m.d[self._r_domain] += produce_r_bin.eq(produce_dec.o)
|
||||||
|
|
||||||
w_full = Signal()
|
w_full = Signal()
|
||||||
r_empty = Signal()
|
r_empty = Signal()
|
||||||
m.d.comb += [
|
m.d.comb += [
|
||||||
|
@ -359,6 +380,9 @@ class AsyncFIFO(Elaboratable, FIFOInterface):
|
||||||
r_empty.eq(consume_r_gry == produce_r_gry),
|
r_empty.eq(consume_r_gry == produce_r_gry),
|
||||||
]
|
]
|
||||||
|
|
||||||
|
m.d[self._w_domain] += self.w_level.eq((produce_w_bin - consume_w_bin)[:self._ctr_bits-1])
|
||||||
|
m.d[self._r_domain] += self.r_level.eq((produce_r_bin - consume_r_bin)[:self._ctr_bits-1])
|
||||||
|
|
||||||
storage = Memory(width=self.width, depth=self.depth)
|
storage = Memory(width=self.width, depth=self.depth)
|
||||||
w_port = m.submodules.w_port = storage.write_port(domain=self._w_domain)
|
w_port = m.submodules.w_port = storage.write_port(domain=self._w_domain)
|
||||||
r_port = m.submodules.r_port = storage.read_port (domain=self._r_domain,
|
r_port = m.submodules.r_port = storage.read_port (domain=self._r_domain,
|
||||||
|
@ -482,6 +506,7 @@ class AsyncFIFOBuffered(Elaboratable, FIFOInterface):
|
||||||
fifo.w_data.eq(self.w_data),
|
fifo.w_data.eq(self.w_data),
|
||||||
self.w_rdy.eq(fifo.w_rdy),
|
self.w_rdy.eq(fifo.w_rdy),
|
||||||
fifo.w_en.eq(self.w_en),
|
fifo.w_en.eq(self.w_en),
|
||||||
|
self.w_level.eq(fifo.w_level),
|
||||||
]
|
]
|
||||||
|
|
||||||
with m.If(self.r_en | ~self.r_rdy):
|
with m.If(self.r_en | ~self.r_rdy):
|
||||||
|
@ -489,6 +514,7 @@ class AsyncFIFOBuffered(Elaboratable, FIFOInterface):
|
||||||
self.r_data.eq(fifo.r_data),
|
self.r_data.eq(fifo.r_data),
|
||||||
self.r_rdy.eq(fifo.r_rdy),
|
self.r_rdy.eq(fifo.r_rdy),
|
||||||
self.r_rst.eq(fifo.r_rst),
|
self.r_rst.eq(fifo.r_rst),
|
||||||
|
self.r_level.eq(fifo.r_level),
|
||||||
]
|
]
|
||||||
m.d.comb += [
|
m.d.comb += [
|
||||||
fifo.r_en.eq(1)
|
fifo.r_en.eq(1)
|
||||||
|
|
|
@ -70,6 +70,8 @@ class FIFOModel(Elaboratable, FIFOInterface):
|
||||||
self.w_domain = w_domain
|
self.w_domain = w_domain
|
||||||
|
|
||||||
self.level = Signal(range(self.depth + 1))
|
self.level = Signal(range(self.depth + 1))
|
||||||
|
self.r_level = Signal(range(self.depth + 1))
|
||||||
|
self.w_level = Signal(range(self.depth + 1))
|
||||||
|
|
||||||
def elaborate(self, platform):
|
def elaborate(self, platform):
|
||||||
m = Module()
|
m = Module()
|
||||||
|
@ -104,6 +106,10 @@ class FIFOModel(Elaboratable, FIFOInterface):
|
||||||
+ (self.w_rdy & self.w_en)
|
+ (self.w_rdy & self.w_en)
|
||||||
- (self.r_rdy & self.r_en))
|
- (self.r_rdy & self.r_en))
|
||||||
|
|
||||||
|
m.d.comb += [
|
||||||
|
self.r_level.eq(self.level),
|
||||||
|
self.w_level.eq(self.level),
|
||||||
|
]
|
||||||
m.d.comb += Assert(ResetSignal(self.r_domain) == ResetSignal(self.w_domain))
|
m.d.comb += Assert(ResetSignal(self.r_domain) == ResetSignal(self.w_domain))
|
||||||
|
|
||||||
return m
|
return m
|
||||||
|
@ -135,8 +141,8 @@ class FIFOModelEquivalenceSpec(Elaboratable):
|
||||||
|
|
||||||
m.d.comb += Assert(dut.r_rdy.implies(gold.r_rdy))
|
m.d.comb += Assert(dut.r_rdy.implies(gold.r_rdy))
|
||||||
m.d.comb += Assert(dut.w_rdy.implies(gold.w_rdy))
|
m.d.comb += Assert(dut.w_rdy.implies(gold.w_rdy))
|
||||||
if hasattr(dut, "level"):
|
m.d.comb += Assert(dut.r_level == gold.r_level)
|
||||||
m.d.comb += Assert(dut.level == gold.level)
|
m.d.comb += Assert(dut.w_level == gold.w_level)
|
||||||
|
|
||||||
if dut.fwft:
|
if dut.fwft:
|
||||||
m.d.comb += Assert(dut.r_rdy
|
m.d.comb += Assert(dut.r_rdy
|
||||||
|
|
Loading…
Reference in a new issue