Skip to content

Commit

Permalink
Tweaks to BasicFifo
Browse files Browse the repository at this point in the history
  • Loading branch information
Lekcyjna committed Nov 19, 2023
1 parent 80cb0cd commit 43ef2df
Show file tree
Hide file tree
Showing 3 changed files with 20 additions and 18 deletions.
2 changes: 2 additions & 0 deletions test/utils/test_fifo.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
from amaranth import *
from amaranth.sim import Settle

from transactron.utils.fifo import BasicFifo
from transactron.lib import AdapterTrans
Expand Down Expand Up @@ -55,6 +56,7 @@ def source():

if random.random() < 0.005:
yield from fifoc.fifo_clear.call()
yield Settle()
expq.clear()

self.done = True
Expand Down
26 changes: 8 additions & 18 deletions transactron/utils/fifo.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,10 +2,12 @@
from transactron import Method, def_method, Priority, TModule
from transactron._utils import MethodLayout
from transactron.utils._typing import ValueLike
from transactron.utils.utils import mod_incr


class BasicFifo(Elaboratable):
"""Transactional FIFO queue
Attributes
----------
read: Method
Expand All @@ -17,6 +19,7 @@ class BasicFifo(Elaboratable):
clear: Method
Clears the FIFO entries. Has priority over `read` and `write` methods.
Note that, clearing the FIFO doesn't reinitialize it to values passed in `init` parameter.
"""

def __init__(self, layout: MethodLayout, depth: int) -> None:
Expand All @@ -28,48 +31,40 @@ def __init__(self, layout: MethodLayout, depth: int) -> None:
If integer is given, Record with field `data` and width of this paramter is used as internal layout.
depth: int
Size of the FIFO.
"""
self.layout = layout
self.width = len(Record(self.layout))
self.depth = depth

self.read = Method(o=self.layout)
self.write = Method(i=self.layout)
self.clear = Method()
self.head = Record(self.layout)

self.buff = Memory(width=self.width, depth=self.depth)

self.write_ready = Signal()
self.read_ready = Signal()

self.read_idx = Signal((self.depth - 1).bit_length())
self.write_idx = Signal((self.depth - 1).bit_length())
# current fifo depth
self.level = Signal((self.depth).bit_length())

self.clear.add_conflict(self.read, Priority.LEFT)
self.clear.add_conflict(self.write, Priority.LEFT)
# for interface compatibility with MultiportFifo
self.read_methods = [self.read]
self.write_methods = [self.write]

def elaborate(self, platform):
def mod_incr(sig: Value, mod: int) -> Value:
# perform (sig+1)%mod operation
if mod == 2 ** len(sig):
return sig + 1
return Mux(sig == mod - 1, 0, sig + 1)

m = TModule()

m.submodules.buff_rdport = self.buff_rdport = self.buff.read_port(
domain="comb", transparent=True
) # FWFT behaviour
next_read_idx = Signal.like(self.read_idx)
m.d.top_comb += next_read_idx.eq(mod_incr(self.read_idx, self.depth))

m.submodules.buff_rdport = self.buff_rdport = self.buff.read_port(domain="sync", transparent=True)
m.submodules.buff_wrport = self.buff_wrport = self.buff.write_port()

m.d.comb += self.read_ready.eq(self.level > 0)
m.d.comb += self.write_ready.eq(self.level < self.depth)
m.d.top_comb += self.read_ready.eq(self.level != 0)
m.d.top_comb += self.write_ready.eq(self.level != self.depth)

Expand All @@ -80,15 +75,11 @@ def mod_incr(sig: Value, mod: int) -> Value:
with m.If(self.clear.run):
m.d.sync += self.level.eq(0)

m.d.comb += self.buff_rdport.addr.eq(self.read_idx)
m.d.comb += self.head.eq(self.buff_rdport.data)
m.d.top_comb += self.buff_rdport.addr.eq(Mux(self.read.run, next_read_idx, self.read_idx))
m.d.top_comb += self.head.eq(self.buff_rdport.data)

@def_method(m, self.write, ready=self.write_ready)
def _(arg: Record) -> None:
m.d.comb += self.buff_wrport.addr.eq(self.write_idx)
m.d.comb += self.buff_wrport.data.eq(arg)
m.d.top_comb += self.buff_wrport.addr.eq(self.write_idx)
m.d.top_comb += self.buff_wrport.data.eq(arg)
m.d.comb += self.buff_wrport.en.eq(1)
Expand All @@ -97,7 +88,6 @@ def _(arg: Record) -> None:

@def_method(m, self.read, self.read_ready)
def _() -> ValueLike:
m.d.sync += self.read_idx.eq(mod_incr(self.read_idx, self.depth))
m.d.sync += self.read_idx.eq(next_read_idx)
return self.head

Expand Down
10 changes: 10 additions & 0 deletions transactron/utils/utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -23,9 +23,19 @@
"popcount",
"count_leading_zeros",
"count_trailing_zeros",
"mod_incr",
]


def mod_incr(sig: Value, mod: int) -> Value:
"""
Perform `(sig+1) % mod` operation.
"""
if mod == 2 ** len(sig):
return sig + 1
return Mux(sig == mod - 1, 0, sig + 1)


@contextmanager
def OneHotSwitch(m: ModuleLike, test: Value):
"""One-hot switch.
Expand Down

0 comments on commit 43ef2df

Please sign in to comment.