2013-05-22 11:11:09 -04:00
|
|
|
from migen.fhdl.std import *
|
2013-02-22 17:19:37 -05:00
|
|
|
from migen.genlib import roundrobin
|
2013-04-01 15:54:21 -04:00
|
|
|
from migen.genlib.record import *
|
2015-06-02 13:29:38 -04:00
|
|
|
from migen.genlib.misc import optree, chooser, FlipFlop, Counter
|
2013-08-26 08:22:54 -04:00
|
|
|
from migen.genlib.fsm import FSM, NextState
|
2012-03-08 12:14:06 -05:00
|
|
|
from migen.bus.transactions import *
|
2011-12-08 12:47:41 -05:00
|
|
|
|
2013-04-01 15:54:21 -04:00
|
|
|
_layout = [
|
2015-04-13 15:33:44 -04:00
|
|
|
("adr", 30, DIR_M_TO_S),
|
|
|
|
("dat_w", "data_width", DIR_M_TO_S),
|
|
|
|
("dat_r", "data_width", DIR_S_TO_M),
|
|
|
|
("sel", "sel_width", DIR_M_TO_S),
|
|
|
|
("cyc", 1, DIR_M_TO_S),
|
|
|
|
("stb", 1, DIR_M_TO_S),
|
|
|
|
("ack", 1, DIR_S_TO_M),
|
|
|
|
("we", 1, DIR_M_TO_S),
|
|
|
|
("cti", 3, DIR_M_TO_S),
|
|
|
|
("bte", 2, DIR_M_TO_S),
|
|
|
|
("err", 1, DIR_S_TO_M)
|
2013-04-01 15:54:21 -04:00
|
|
|
]
|
2011-12-08 12:47:41 -05:00
|
|
|
|
2015-04-13 14:45:35 -04:00
|
|
|
|
2013-04-01 15:54:21 -04:00
|
|
|
class Interface(Record):
|
2015-04-13 14:07:07 -04:00
|
|
|
def __init__(self, data_width=32):
|
|
|
|
Record.__init__(self, set_layout_parameters(_layout,
|
|
|
|
data_width=data_width,
|
|
|
|
sel_width=data_width//8))
|
2011-12-08 12:47:41 -05:00
|
|
|
|
2015-04-13 14:45:35 -04:00
|
|
|
|
2013-04-01 15:54:21 -04:00
|
|
|
class InterconnectPointToPoint(Module):
|
2015-04-13 14:07:07 -04:00
|
|
|
def __init__(self, master, slave):
|
|
|
|
self.comb += master.connect(slave)
|
2011-12-08 17:21:25 -05:00
|
|
|
|
2015-04-13 14:45:35 -04:00
|
|
|
|
2013-04-01 15:54:21 -04:00
|
|
|
class Arbiter(Module):
|
2015-04-13 14:07:07 -04:00
|
|
|
def __init__(self, masters, target):
|
|
|
|
self.submodules.rr = roundrobin.RoundRobin(len(masters))
|
|
|
|
|
|
|
|
# mux master->slave signals
|
|
|
|
for name, size, direction in _layout:
|
|
|
|
if direction == DIR_M_TO_S:
|
|
|
|
choices = Array(getattr(m, name) for m in masters)
|
|
|
|
self.comb += getattr(target, name).eq(choices[self.rr.grant])
|
|
|
|
|
|
|
|
# connect slave->master signals
|
|
|
|
for name, size, direction in _layout:
|
|
|
|
if direction == DIR_S_TO_M:
|
|
|
|
source = getattr(target, name)
|
|
|
|
for i, m in enumerate(masters):
|
|
|
|
dest = getattr(m, name)
|
|
|
|
if name == "ack" or name == "err":
|
|
|
|
self.comb += dest.eq(source & (self.rr.grant == i))
|
|
|
|
else:
|
|
|
|
self.comb += dest.eq(source)
|
|
|
|
|
|
|
|
# connect bus requests to round-robin selector
|
|
|
|
reqs = [m.cyc for m in masters]
|
|
|
|
self.comb += self.rr.request.eq(Cat(*reqs))
|
2011-12-08 17:21:25 -05:00
|
|
|
|
2015-04-13 14:45:35 -04:00
|
|
|
|
2013-04-01 15:54:21 -04:00
|
|
|
class Decoder(Module):
|
2015-04-13 14:07:07 -04:00
|
|
|
# slaves is a list of pairs:
|
|
|
|
# 0) function that takes the address signal and returns a FHDL expression
|
|
|
|
# that evaluates to 1 when the slave is selected and 0 otherwise.
|
|
|
|
# 1) wishbone.Slave reference.
|
|
|
|
# register adds flip-flops after the address comparators. Improves timing,
|
|
|
|
# but breaks Wishbone combinatorial feedback.
|
|
|
|
def __init__(self, master, slaves, register=False):
|
|
|
|
ns = len(slaves)
|
|
|
|
slave_sel = Signal(ns)
|
|
|
|
slave_sel_r = Signal(ns)
|
|
|
|
|
|
|
|
# decode slave addresses
|
|
|
|
self.comb += [slave_sel[i].eq(fun(master.adr))
|
|
|
|
for i, (fun, bus) in enumerate(slaves)]
|
|
|
|
if register:
|
|
|
|
self.sync += slave_sel_r.eq(slave_sel)
|
|
|
|
else:
|
|
|
|
self.comb += slave_sel_r.eq(slave_sel)
|
|
|
|
|
|
|
|
# connect master->slaves signals except cyc
|
|
|
|
for slave in slaves:
|
|
|
|
for name, size, direction in _layout:
|
|
|
|
if direction == DIR_M_TO_S and name != "cyc":
|
|
|
|
self.comb += getattr(slave[1], name).eq(getattr(master, name))
|
|
|
|
|
|
|
|
# combine cyc with slave selection signals
|
|
|
|
self.comb += [slave[1].cyc.eq(master.cyc & slave_sel[i])
|
|
|
|
for i, slave in enumerate(slaves)]
|
|
|
|
|
|
|
|
# generate master ack (resp. err) by ORing all slave acks (resp. errs)
|
|
|
|
self.comb += [
|
|
|
|
master.ack.eq(optree("|", [slave[1].ack for slave in slaves])),
|
|
|
|
master.err.eq(optree("|", [slave[1].err for slave in slaves]))
|
|
|
|
]
|
|
|
|
|
|
|
|
# mux (1-hot) slave data return
|
|
|
|
masked = [Replicate(slave_sel_r[i], flen(master.dat_r)) & slaves[i][1].dat_r for i in range(ns)]
|
|
|
|
self.comb += master.dat_r.eq(optree("|", masked))
|
2011-12-09 07:11:52 -05:00
|
|
|
|
2015-04-13 14:45:35 -04:00
|
|
|
|
2013-04-01 15:54:21 -04:00
|
|
|
class InterconnectShared(Module):
|
2015-04-13 14:07:07 -04:00
|
|
|
def __init__(self, masters, slaves, register=False):
|
|
|
|
shared = Interface()
|
|
|
|
self.submodules += Arbiter(masters, shared)
|
|
|
|
self.submodules += Decoder(shared, slaves, register)
|
2012-03-08 12:14:06 -05:00
|
|
|
|
2015-04-13 14:45:35 -04:00
|
|
|
|
2013-07-22 04:12:50 -04:00
|
|
|
class Crossbar(Module):
|
2015-04-13 14:07:07 -04:00
|
|
|
def __init__(self, masters, slaves, register=False):
|
|
|
|
matches, busses = zip(*slaves)
|
|
|
|
access = [[Interface() for j in slaves] for i in masters]
|
|
|
|
# decode each master into its access row
|
|
|
|
for row, master in zip(access, masters):
|
|
|
|
row = list(zip(matches, row))
|
|
|
|
self.submodules += Decoder(master, row, register)
|
|
|
|
# arbitrate each access column onto its slave
|
|
|
|
for column, bus in zip(zip(*access), busses):
|
|
|
|
self.submodules += Arbiter(column, bus)
|
2013-07-22 04:12:50 -04:00
|
|
|
|
2015-04-13 14:45:35 -04:00
|
|
|
|
2013-08-26 08:22:54 -04:00
|
|
|
class DownConverter(Module):
|
2015-06-02 13:29:38 -04:00
|
|
|
"""DownConverter
|
|
|
|
|
|
|
|
This module splits Wishbone accesses from a master interface to a smaller
|
|
|
|
slave interface.
|
|
|
|
|
|
|
|
Writes:
|
|
|
|
Writes from master are splitted N writes to the slave. Access is acked when the last
|
|
|
|
access is acked by the slave.
|
|
|
|
|
|
|
|
Reads:
|
|
|
|
Read from master are splitted in N reads to the the slave. Read datas from
|
|
|
|
the slave are cached before being presented concatenated on the last access.
|
|
|
|
|
|
|
|
TODO:
|
|
|
|
Manage err signal? (Not implemented since we generally don't use it on Migen/MiSoC modules)
|
|
|
|
"""
|
|
|
|
def __init__(self, master, slave):
|
|
|
|
dw_from = flen(master.dat_r)
|
|
|
|
dw_to = flen(slave.dat_w)
|
|
|
|
ratio = dw_from//dw_to
|
|
|
|
|
|
|
|
# # #
|
|
|
|
|
|
|
|
read = Signal()
|
|
|
|
write = Signal()
|
|
|
|
|
|
|
|
counter = Counter(max=ratio)
|
|
|
|
self.submodules += counter
|
|
|
|
counter_done = Signal()
|
|
|
|
self.comb += counter_done.eq(counter.value == ratio-1)
|
|
|
|
|
|
|
|
# Main FSM
|
|
|
|
self.submodules.fsm = fsm = FSM(reset_state="IDLE")
|
|
|
|
fsm.act("IDLE",
|
|
|
|
counter.reset.eq(1),
|
|
|
|
If(master.stb & master.cyc,
|
|
|
|
If(master.we,
|
|
|
|
NextState("WRITE")
|
|
|
|
).Else(
|
|
|
|
NextState("READ")
|
|
|
|
)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
fsm.act("WRITE",
|
|
|
|
write.eq(1),
|
|
|
|
slave.we.eq(1),
|
|
|
|
slave.cyc.eq(1),
|
|
|
|
If(master.stb & master.cyc,
|
|
|
|
slave.stb.eq(1),
|
|
|
|
If(slave.ack,
|
|
|
|
counter.ce.eq(1),
|
|
|
|
If(counter_done,
|
|
|
|
master.ack.eq(1),
|
|
|
|
NextState("IDLE")
|
|
|
|
)
|
|
|
|
)
|
|
|
|
).Elif(~master.cyc,
|
|
|
|
NextState("IDLE")
|
|
|
|
)
|
|
|
|
)
|
|
|
|
fsm.act("READ",
|
|
|
|
read.eq(1),
|
|
|
|
slave.cyc.eq(1),
|
|
|
|
If(master.stb & master.cyc,
|
|
|
|
slave.stb.eq(1),
|
|
|
|
If(slave.ack,
|
|
|
|
counter.ce.eq(1),
|
|
|
|
If(counter_done,
|
|
|
|
master.ack.eq(1),
|
|
|
|
NextState("IDLE")
|
|
|
|
)
|
|
|
|
)
|
|
|
|
).Elif(~master.cyc,
|
|
|
|
NextState("IDLE")
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
# Address
|
2015-04-13 14:07:07 -04:00
|
|
|
self.comb += [
|
2015-06-02 13:29:38 -04:00
|
|
|
If(counter_done,
|
|
|
|
slave.cti.eq(7) # indicate end of burst
|
|
|
|
).Else(
|
|
|
|
slave.cti.eq(2)
|
|
|
|
),
|
|
|
|
slave.adr.eq(Cat(counter.value, master.adr))
|
2015-04-13 14:07:07 -04:00
|
|
|
]
|
|
|
|
|
2015-06-02 13:29:38 -04:00
|
|
|
# Datapath
|
|
|
|
cases = {}
|
|
|
|
for i in range(ratio):
|
|
|
|
cases[i] = [
|
|
|
|
slave.sel.eq(master.sel[i*dw_to//8:(i+1)*dw_to]),
|
|
|
|
slave.dat_w.eq(master.dat_w[i*dw_to:(i+1)*dw_to])
|
|
|
|
]
|
|
|
|
self.comb += Case(counter.value, cases)
|
|
|
|
|
|
|
|
|
|
|
|
cached_data = Signal(dw_from)
|
|
|
|
self.comb += master.dat_r.eq(Cat(cached_data[dw_to:], slave.dat_r))
|
|
|
|
self.sync += \
|
|
|
|
If(read & counter.ce,
|
|
|
|
cached_data.eq(master.dat_r)
|
|
|
|
)
|
|
|
|
|
2015-04-13 14:07:07 -04:00
|
|
|
|
2015-06-02 13:29:38 -04:00
|
|
|
class UpConverter(Module):
|
|
|
|
"""UpConverter
|
2015-04-13 14:07:07 -04:00
|
|
|
|
2015-06-02 13:29:38 -04:00
|
|
|
This module up-converts wishbone accesses and bursts from a master interface
|
|
|
|
to a wider slave interface. This allows efficient use wishbone bursts.
|
2015-04-13 14:07:07 -04:00
|
|
|
|
2015-06-02 13:29:38 -04:00
|
|
|
Writes:
|
|
|
|
Wishbone writes are cached before being written to the slave. Access to
|
|
|
|
the slave is done at the end of a burst or when address reach end of burst
|
|
|
|
addressing.
|
2015-04-13 14:07:07 -04:00
|
|
|
|
2015-06-02 13:29:38 -04:00
|
|
|
Reads:
|
|
|
|
Cache is refilled only at the beginning of each burst, the subsequent
|
|
|
|
reads of a burst use the cached data.
|
2015-04-13 14:07:07 -04:00
|
|
|
|
2015-06-02 13:29:38 -04:00
|
|
|
TODO:
|
|
|
|
Manage err signal? (Not implemented since we generally don't use it on Migen/MiSoC modules)
|
|
|
|
"""
|
|
|
|
def __init__(self, master, slave):
|
|
|
|
dw_from = flen(master.dat_r)
|
|
|
|
dw_to = flen(slave.dat_w)
|
|
|
|
ratio = dw_to//dw_from
|
|
|
|
ratiobits = log2_int(ratio)
|
|
|
|
|
|
|
|
# # #
|
|
|
|
|
|
|
|
write = Signal()
|
|
|
|
evict = Signal()
|
|
|
|
refill = Signal()
|
|
|
|
read = Signal()
|
|
|
|
|
|
|
|
address = FlipFlop(30)
|
|
|
|
self.submodules += address
|
|
|
|
self.comb += address.d.eq(master.adr)
|
|
|
|
|
|
|
|
counter = Counter(max=ratio)
|
|
|
|
self.submodules += counter
|
|
|
|
counter_offset = Signal(max=ratio)
|
|
|
|
counter_done = Signal()
|
2015-04-13 14:07:07 -04:00
|
|
|
self.comb += [
|
2015-06-02 13:29:38 -04:00
|
|
|
counter_offset.eq(address.q),
|
|
|
|
counter_done.eq((counter.value + counter_offset) == ratio-1)
|
2015-04-13 14:07:07 -04:00
|
|
|
]
|
|
|
|
|
2015-06-02 13:29:38 -04:00
|
|
|
cached_data = Signal(dw_to)
|
|
|
|
cached_sel = Signal(dw_to//8)
|
2015-04-13 14:07:07 -04:00
|
|
|
|
2015-06-02 13:29:38 -04:00
|
|
|
end_of_burst = Signal()
|
|
|
|
self.comb += end_of_burst.eq(~master.cyc |
|
|
|
|
(master.stb & master.cyc & master.ack & ((master.cti == 7) | counter_done)))
|
2015-04-13 14:07:07 -04:00
|
|
|
|
2015-06-02 13:29:38 -04:00
|
|
|
|
|
|
|
need_refill = FlipFlop(reset=1)
|
|
|
|
self.submodules += need_refill
|
|
|
|
self.comb += [
|
|
|
|
need_refill.reset.eq(end_of_burst),
|
|
|
|
need_refill.d.eq(0)
|
|
|
|
]
|
|
|
|
|
|
|
|
# Main FSM
|
|
|
|
self.submodules.fsm = fsm = FSM()
|
2015-04-13 14:07:07 -04:00
|
|
|
fsm.act("IDLE",
|
2015-06-02 13:29:38 -04:00
|
|
|
counter.reset.eq(1),
|
|
|
|
If(master.stb & master.cyc,
|
|
|
|
address.ce.eq(1),
|
|
|
|
If(master.we,
|
|
|
|
NextState("WRITE")
|
|
|
|
).Else(
|
|
|
|
If(need_refill.q,
|
|
|
|
NextState("REFILL")
|
|
|
|
).Else(
|
|
|
|
NextState("READ")
|
|
|
|
)
|
|
|
|
)
|
|
|
|
)
|
2015-04-13 14:07:07 -04:00
|
|
|
)
|
2015-06-02 13:29:38 -04:00
|
|
|
fsm.act("WRITE",
|
|
|
|
If(master.stb & master.cyc,
|
|
|
|
write.eq(1),
|
|
|
|
counter.ce.eq(1),
|
|
|
|
master.ack.eq(1),
|
|
|
|
If(counter_done,
|
|
|
|
NextState("EVICT")
|
|
|
|
)
|
|
|
|
).Elif(~master.cyc,
|
|
|
|
NextState("EVICT")
|
2015-04-13 14:07:07 -04:00
|
|
|
)
|
|
|
|
)
|
2015-06-02 13:29:38 -04:00
|
|
|
fsm.act("EVICT",
|
|
|
|
evict.eq(1),
|
|
|
|
slave.stb.eq(1),
|
|
|
|
slave.we.eq(1),
|
|
|
|
slave.cyc.eq(1),
|
|
|
|
slave.dat_w.eq(cached_data),
|
|
|
|
slave.sel.eq(cached_sel),
|
|
|
|
If(slave.ack,
|
|
|
|
NextState("IDLE")
|
2015-04-13 14:07:07 -04:00
|
|
|
)
|
|
|
|
)
|
2015-06-02 13:29:38 -04:00
|
|
|
fsm.act("REFILL",
|
|
|
|
refill.eq(1),
|
|
|
|
slave.stb.eq(1),
|
|
|
|
slave.cyc.eq(1),
|
|
|
|
If(slave.ack,
|
|
|
|
need_refill.ce.eq(1),
|
|
|
|
NextState("READ")
|
|
|
|
)
|
|
|
|
)
|
|
|
|
fsm.act("READ",
|
|
|
|
read.eq(1),
|
|
|
|
If(master.stb & master.cyc,
|
|
|
|
master.ack.eq(1)
|
|
|
|
),
|
|
|
|
NextState("IDLE")
|
|
|
|
)
|
|
|
|
|
|
|
|
# Address
|
|
|
|
self.comb += [
|
|
|
|
slave.cti.eq(7), # we are not able to generate bursts since up-converting
|
|
|
|
slave.adr.eq(address.q[ratiobits:])
|
|
|
|
]
|
|
|
|
|
|
|
|
# Datapath
|
|
|
|
cached_datas = [FlipFlop(dw_from) for i in range(ratio)]
|
|
|
|
cached_sels = [FlipFlop(dw_from//8) for i in range(ratio)]
|
|
|
|
self.submodules += cached_datas, cached_sels
|
|
|
|
|
|
|
|
cases = {}
|
|
|
|
for i in range(ratio):
|
|
|
|
write_sel = Signal()
|
|
|
|
cases[i] = write_sel.eq(1)
|
|
|
|
self.comb += [
|
|
|
|
cached_sels[i].reset.eq(counter.reset),
|
|
|
|
If(write,
|
|
|
|
cached_datas[i].d.eq(master.dat_w),
|
|
|
|
).Else(
|
|
|
|
cached_datas[i].d.eq(slave.dat_r[dw_from*i:dw_from*(i+1)])
|
|
|
|
),
|
|
|
|
cached_sels[i].d.eq(master.sel),
|
|
|
|
If((write & write_sel) | refill,
|
|
|
|
cached_datas[i].ce.eq(1),
|
|
|
|
cached_sels[i].ce.eq(1)
|
|
|
|
)
|
|
|
|
]
|
|
|
|
self.comb += Case(counter.value + counter_offset, cases)
|
|
|
|
|
|
|
|
cases = {}
|
|
|
|
for i in range(ratio):
|
|
|
|
cases[i] = master.dat_r.eq(cached_datas[i].q)
|
|
|
|
self.comb += Case(address.q[:ratiobits], cases)
|
|
|
|
|
|
|
|
self.comb += [
|
|
|
|
cached_data.eq(Cat([cached_data.q for cached_data in cached_datas])),
|
|
|
|
cached_sel.eq(Cat([cached_sel.q for cached_sel in cached_sels]))
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
class Converter(Module):
|
|
|
|
"""Converter
|
|
|
|
|
|
|
|
This module is a wrapper for DownConverter and UpConverter.
|
|
|
|
It should preferably be used rather than direct instantiations
|
|
|
|
of specific converters.
|
|
|
|
"""
|
|
|
|
def __init__(self, master, slave):
|
|
|
|
self.master = master
|
|
|
|
self.slave = slave
|
|
|
|
|
|
|
|
# # #
|
|
|
|
|
|
|
|
dw_from = flen(master.dat_r)
|
|
|
|
dw_to = flen(slave.dat_r)
|
|
|
|
if dw_from > dw_to:
|
|
|
|
downconverter = DownConverter(master, slave)
|
|
|
|
self.submodules += downconverter
|
|
|
|
elif dw_from < dw_to:
|
|
|
|
upconverter = UpConverter(master, slave)
|
|
|
|
self.submodules += upconverter
|
|
|
|
else:
|
|
|
|
Record.connect(master, slave)
|
2013-08-26 08:22:54 -04:00
|
|
|
|
2015-04-13 14:45:35 -04:00
|
|
|
|
2013-03-15 14:41:30 -04:00
|
|
|
class Tap(Module):
|
2015-04-13 14:07:07 -04:00
|
|
|
def __init__(self, bus, handler=print):
|
|
|
|
self.bus = bus
|
|
|
|
self.handler = handler
|
|
|
|
|
|
|
|
def do_simulation(self, selfp):
|
|
|
|
if selfp.bus.ack:
|
|
|
|
assert(selfp.bus.cyc and selfp.bus.stb)
|
|
|
|
if selfp.bus.we:
|
|
|
|
transaction = TWrite(selfp.bus.adr,
|
|
|
|
selfp.bus.dat_w,
|
|
|
|
selfp.bus.sel)
|
|
|
|
else:
|
|
|
|
transaction = TRead(selfp.bus.adr,
|
|
|
|
selfp.bus.dat_r)
|
|
|
|
self.handler(transaction)
|
|
|
|
do_simulation.passive = True
|
2012-03-08 12:14:06 -05:00
|
|
|
|
2015-04-13 14:45:35 -04:00
|
|
|
|
2013-03-15 14:41:30 -04:00
|
|
|
class Initiator(Module):
|
2015-04-13 14:07:07 -04:00
|
|
|
def __init__(self, generator, bus=None):
|
|
|
|
self.generator = generator
|
|
|
|
if bus is None:
|
|
|
|
bus = Interface()
|
|
|
|
self.bus = bus
|
|
|
|
self.transaction_start = 0
|
|
|
|
self.transaction = None
|
|
|
|
|
|
|
|
def do_simulation(self, selfp):
|
|
|
|
if self.transaction is None or selfp.bus.ack:
|
|
|
|
if self.transaction is not None:
|
|
|
|
self.transaction.latency = selfp.simulator.cycle_counter - self.transaction_start - 1
|
|
|
|
if isinstance(self.transaction, TRead):
|
|
|
|
self.transaction.data = selfp.bus.dat_r
|
|
|
|
try:
|
|
|
|
self.transaction = next(self.generator)
|
|
|
|
except StopIteration:
|
|
|
|
selfp.bus.cyc = 0
|
|
|
|
selfp.bus.stb = 0
|
|
|
|
raise StopSimulation
|
|
|
|
if self.transaction is not None:
|
|
|
|
self.transaction_start = selfp.simulator.cycle_counter
|
|
|
|
selfp.bus.cyc = 1
|
|
|
|
selfp.bus.stb = 1
|
|
|
|
selfp.bus.adr = self.transaction.address
|
|
|
|
if isinstance(self.transaction, TWrite):
|
|
|
|
selfp.bus.we = 1
|
|
|
|
selfp.bus.sel = self.transaction.sel
|
|
|
|
selfp.bus.dat_w = self.transaction.data
|
|
|
|
else:
|
|
|
|
selfp.bus.we = 0
|
|
|
|
else:
|
|
|
|
selfp.bus.cyc = 0
|
|
|
|
selfp.bus.stb = 0
|
2012-06-10 10:40:33 -04:00
|
|
|
|
2015-04-13 14:45:35 -04:00
|
|
|
|
2012-06-10 11:05:10 -04:00
|
|
|
class TargetModel:
|
2015-04-13 14:07:07 -04:00
|
|
|
def read(self, address):
|
|
|
|
return 0
|
2014-10-17 05:08:37 -04:00
|
|
|
|
2015-04-13 14:07:07 -04:00
|
|
|
def write(self, address, data, sel):
|
|
|
|
pass
|
2014-10-17 05:08:37 -04:00
|
|
|
|
2015-04-13 14:07:07 -04:00
|
|
|
def can_ack(self, bus):
|
|
|
|
return True
|
2012-06-10 11:05:10 -04:00
|
|
|
|
2015-04-13 14:45:35 -04:00
|
|
|
|
2013-03-15 14:41:30 -04:00
|
|
|
class Target(Module):
|
2015-04-13 14:07:07 -04:00
|
|
|
def __init__(self, model, bus=None):
|
|
|
|
if bus is None:
|
|
|
|
bus = Interface()
|
|
|
|
self.bus = bus
|
|
|
|
self.model = model
|
|
|
|
|
|
|
|
def do_simulation(self, selfp):
|
|
|
|
bus = selfp.bus
|
|
|
|
if not bus.ack:
|
|
|
|
if self.model.can_ack(bus) and bus.cyc and bus.stb:
|
|
|
|
if bus.we:
|
|
|
|
self.model.write(bus.adr, bus.dat_w, bus.sel)
|
|
|
|
else:
|
|
|
|
bus.dat_r = self.model.read(bus.adr)
|
|
|
|
bus.ack = 1
|
|
|
|
else:
|
|
|
|
bus.ack = 0
|
|
|
|
do_simulation.passive = True
|
2012-12-01 06:59:09 -05:00
|
|
|
|
2015-04-13 14:45:35 -04:00
|
|
|
|
2013-04-01 15:54:21 -04:00
|
|
|
class SRAM(Module):
|
2015-04-13 14:07:07 -04:00
|
|
|
def __init__(self, mem_or_size, read_only=None, init=None, bus=None):
|
|
|
|
if bus is None:
|
|
|
|
bus = Interface()
|
|
|
|
self.bus = bus
|
|
|
|
bus_data_width = flen(self.bus.dat_r)
|
|
|
|
if isinstance(mem_or_size, Memory):
|
|
|
|
assert(mem_or_size.width <= bus_data_width)
|
|
|
|
self.mem = mem_or_size
|
|
|
|
else:
|
|
|
|
self.mem = Memory(bus_data_width, mem_or_size//(bus_data_width//8), init=init)
|
|
|
|
if read_only is None:
|
|
|
|
if hasattr(self.mem, "bus_read_only"):
|
|
|
|
read_only = self.mem.bus_read_only
|
|
|
|
else:
|
|
|
|
read_only = False
|
|
|
|
|
|
|
|
###
|
|
|
|
|
|
|
|
# memory
|
|
|
|
port = self.mem.get_port(write_capable=not read_only, we_granularity=8)
|
|
|
|
self.specials += self.mem, port
|
|
|
|
# generate write enable signal
|
|
|
|
if not read_only:
|
|
|
|
self.comb += [port.we[i].eq(self.bus.cyc & self.bus.stb & self.bus.we & self.bus.sel[i])
|
|
|
|
for i in range(4)]
|
|
|
|
# address and data
|
|
|
|
self.comb += [
|
|
|
|
port.adr.eq(self.bus.adr[:flen(port.adr)]),
|
|
|
|
self.bus.dat_r.eq(port.dat_r)
|
|
|
|
]
|
|
|
|
if not read_only:
|
|
|
|
self.comb += port.dat_w.eq(self.bus.dat_w),
|
|
|
|
# generate ack
|
|
|
|
self.sync += [
|
|
|
|
self.bus.ack.eq(0),
|
|
|
|
If(self.bus.cyc & self.bus.stb & ~self.bus.ack, self.bus.ack.eq(1))
|
|
|
|
]
|