forked from M-Labs/artiq-zynq
sim: prototyping frame decoding pipeine
sim: add double buffer sim: add eop marker for crc checker in double buffer sim: add KCode, pak type & CRC generator sim: add Stream crossbar sim: add stream pipeline with parser & buffer sim: add frame generator & image viewer sim: add arbiter sim: add broadcaster, double buffer & eop tester
This commit is contained in:
parent
3c44c66ab2
commit
f6b13f271d
149
sim_arbiter.py
Normal file
149
sim_arbiter.py
Normal file
@ -0,0 +1,149 @@
|
|||||||
|
from migen import *
|
||||||
|
from misoc.interconnect.csr import *
|
||||||
|
from misoc.interconnect import stream
|
||||||
|
|
||||||
|
from sim_generator import CXPCRC32Inserter
|
||||||
|
from src.gateware.cxp_frame_pipeline import *
|
||||||
|
from src.gateware.cxp_pipeline import *
|
||||||
|
|
||||||
|
CXP_CHANNELS = 2
|
||||||
|
|
||||||
|
|
||||||
|
class Frame_Pipeline(Module):
|
||||||
|
def __init__(self, n_downconn):
|
||||||
|
# to construct correct crc and ack/stb signal
|
||||||
|
self.submodules.arbiter = arbiter = Stream_Arbiter(n_downconn)
|
||||||
|
self.submodules.broadcaster = broadcaster = Stream_Broadcaster(1)
|
||||||
|
self.submodules.buffer = buffer = Buffer(word_layout_dchar)
|
||||||
|
self.sinks = []
|
||||||
|
|
||||||
|
for i in range(n_downconn):
|
||||||
|
# generating the packet
|
||||||
|
dchar_decoder = Duplicated_Char_Decoder()
|
||||||
|
eop_marker = EOP_Marker()
|
||||||
|
|
||||||
|
pipeline = [dchar_decoder, eop_marker]
|
||||||
|
self.submodules += pipeline
|
||||||
|
|
||||||
|
for s, d in zip(pipeline, pipeline[1:]):
|
||||||
|
self.comb += s.source.connect(d.sink)
|
||||||
|
self.sinks.append(pipeline[0].sink)
|
||||||
|
|
||||||
|
self.comb += pipeline[-1].source.connect(arbiter.sinks[i])
|
||||||
|
|
||||||
|
# self.comb += arbiter.source.ack.eq(1)
|
||||||
|
|
||||||
|
self.comb += [
|
||||||
|
arbiter.source.connect(broadcaster.sink),
|
||||||
|
broadcaster.sources[0].connect(buffer.sink),
|
||||||
|
]
|
||||||
|
self.comb += buffer.source.ack.eq(1)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
dut = Frame_Pipeline(CXP_CHANNELS)
|
||||||
|
|
||||||
|
|
||||||
|
def packet_sim(packets=[], active_ch=2):
|
||||||
|
assert active_ch <= CXP_CHANNELS
|
||||||
|
|
||||||
|
print("=================TEST========================")
|
||||||
|
# yield dut.arbiter.active_channels.eq((2**active_ch) - 1)
|
||||||
|
yield dut.arbiter.active_channels.eq((2**active_ch) - 1)
|
||||||
|
sinks = dut.sinks
|
||||||
|
|
||||||
|
for p in packets:
|
||||||
|
for i in range(CXP_CHANNELS):
|
||||||
|
if p["sink_id"] == i:
|
||||||
|
yield sinks[p["sink_id"]].data.eq(p["data"])
|
||||||
|
yield sinks[p["sink_id"]].k.eq(p["k"])
|
||||||
|
yield sinks[p["sink_id"]].stb.eq(1)
|
||||||
|
if "eop" in p:
|
||||||
|
yield sinks[p["sink_id"]].eop.eq(1)
|
||||||
|
else:
|
||||||
|
yield sinks[p["sink_id"]].eop.eq(0)
|
||||||
|
else:
|
||||||
|
yield sinks[i].data.eq(0)
|
||||||
|
yield sinks[i].k.eq(0)
|
||||||
|
yield sinks[i].stb.eq(0)
|
||||||
|
yield sinks[i].eop.eq(0)
|
||||||
|
yield
|
||||||
|
|
||||||
|
# extra clk cycles
|
||||||
|
for _ in range(100):
|
||||||
|
for i in range(CXP_CHANNELS):
|
||||||
|
yield sinks[i].data.eq(0)
|
||||||
|
yield sinks[i].k.eq(0)
|
||||||
|
yield sinks[i].stb.eq(0)
|
||||||
|
yield sinks[i].eop.eq(0)
|
||||||
|
yield
|
||||||
|
assert True
|
||||||
|
|
||||||
|
|
||||||
|
def testbench(n_downconn, with_header=True, with_crc=True):
|
||||||
|
paks = []
|
||||||
|
|
||||||
|
stream_id = 0
|
||||||
|
pix_len = 10
|
||||||
|
for p in range(20):
|
||||||
|
n_conn = p % n_downconn
|
||||||
|
if with_header:
|
||||||
|
frame_pak_len = pix_len if with_crc else pix_len - 1
|
||||||
|
frame = [
|
||||||
|
{
|
||||||
|
"sink_id": n_conn,
|
||||||
|
"data": Replicate(C(stream_id, char_width), 4),
|
||||||
|
"k": Replicate(0, 4),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"sink_id": n_conn,
|
||||||
|
"data": Replicate(C(0, char_width), 4),
|
||||||
|
"k": Replicate(0, 4),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"sink_id": n_conn,
|
||||||
|
"data": Replicate(C(frame_pak_len, 2 * char_width)[8:], 4),
|
||||||
|
"k": Replicate(0, 4),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"sink_id": n_conn,
|
||||||
|
"data": Replicate(C(frame_pak_len, 2 * char_width)[:8], 4),
|
||||||
|
"k": Replicate(0, 4),
|
||||||
|
},
|
||||||
|
]
|
||||||
|
else:
|
||||||
|
frame = []
|
||||||
|
|
||||||
|
for n_pix in range(pix_len):
|
||||||
|
if n_pix < pix_len - 1:
|
||||||
|
frame.append(
|
||||||
|
{
|
||||||
|
"sink_id": n_conn,
|
||||||
|
# "data": C((n_conn + 1) << 28 | p << 24 | n_pix, word_width),
|
||||||
|
"data": C( p << 24 | n_pix, word_width),
|
||||||
|
"k": Replicate(0, 4),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
frame.append(
|
||||||
|
{
|
||||||
|
"sink_id": n_conn,
|
||||||
|
# "data": C((n_conn + 1) << 28 | p << 24 | n_pix, word_width),
|
||||||
|
"data": C( p << 24 | n_pix, word_width),
|
||||||
|
"k": Replicate(0, 4),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
frame.append(
|
||||||
|
{
|
||||||
|
"sink_id": n_conn,
|
||||||
|
"data": Replicate(KCode["pak_end"], 4),
|
||||||
|
"k": Replicate(1, 4),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
paks += frame
|
||||||
|
|
||||||
|
yield from packet_sim(paks)
|
||||||
|
|
||||||
|
|
||||||
|
run_simulation(dut, testbench(CXP_CHANNELS,with_crc=False), vcd_name="sim-cxp.vcd")
|
91
sim_broadcaster.py
Normal file
91
sim_broadcaster.py
Normal file
@ -0,0 +1,91 @@
|
|||||||
|
from migen import *
|
||||||
|
from misoc.interconnect.csr import *
|
||||||
|
from misoc.interconnect import stream
|
||||||
|
|
||||||
|
from sim_generator import CXPCRC32Inserter
|
||||||
|
from src.gateware.cxp_frame_pipeline import *
|
||||||
|
from src.gateware.cxp_pipeline import *
|
||||||
|
|
||||||
|
class double_buffer_pipeline(Module):
|
||||||
|
def __init__(self):
|
||||||
|
fifo = stream.SyncFIFO(word_layout, 32)
|
||||||
|
dchar_decoder = Duplicated_Char_Decoder()
|
||||||
|
broadcaster = Stream_Broadcaster(1)
|
||||||
|
pipeline = [fifo, dchar_decoder, broadcaster]
|
||||||
|
self.submodules += pipeline
|
||||||
|
|
||||||
|
for s, d in zip(pipeline, pipeline[1:]):
|
||||||
|
self.comb += s.source.connect(d.sink)
|
||||||
|
|
||||||
|
self.sink = pipeline[0].sink
|
||||||
|
|
||||||
|
self.submodules.buffer = buffer = Buffer(word_layout_dchar)
|
||||||
|
self.comb += broadcaster.sources[0].connect(buffer.sink)
|
||||||
|
self.source = buffer.source
|
||||||
|
|
||||||
|
|
||||||
|
# for sim, no backpressure
|
||||||
|
self.comb += self.source.ack.eq(1)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
dut = double_buffer_pipeline()
|
||||||
|
|
||||||
|
|
||||||
|
def packet_sim(packets=[]):
|
||||||
|
print("=================TEST========================")
|
||||||
|
sink = dut.sink
|
||||||
|
|
||||||
|
cyc = len(packets)
|
||||||
|
pak = packets
|
||||||
|
for c in range(cyc):
|
||||||
|
yield sink.data.eq(pak[c]["data"])
|
||||||
|
yield sink.k.eq(pak[c]["k"])
|
||||||
|
yield sink.stb.eq(1)
|
||||||
|
if "eop" in pak[c]:
|
||||||
|
yield sink.eop.eq(1)
|
||||||
|
else:
|
||||||
|
yield sink.eop.eq(0)
|
||||||
|
yield
|
||||||
|
|
||||||
|
# extra clk cycles
|
||||||
|
for _ in range(cyc, cyc + 20):
|
||||||
|
yield sink.data.eq(0)
|
||||||
|
yield sink.k.eq(0)
|
||||||
|
yield sink.stb.eq(0)
|
||||||
|
yield sink.eop.eq(0)
|
||||||
|
yield
|
||||||
|
assert True
|
||||||
|
|
||||||
|
|
||||||
|
def testbench():
|
||||||
|
paks = [
|
||||||
|
{"data": Replicate(C(0, char_width), 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(C(0, char_width), 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(C(4, 2 * char_width)[8:], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(C(4, 2 * char_width)[:8], 4), "k": Replicate(0, 4)}, # CRC doesn't count
|
||||||
|
{"data": C(0x7C7C7C7C, word_width), "k": Replicate(1, 4)},
|
||||||
|
{"data": C(0x01010101, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0xF6ACEF6A, word_width), "k": Replicate(0, 4)},
|
||||||
|
|
||||||
|
|
||||||
|
{"data": Replicate(C(0, char_width), 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(C(1, char_width), 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(C(8, 2 * char_width)[8:], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(C(8, 2 * char_width)[:8], 4), "k": Replicate(0, 4)}, # CRC doesn't count
|
||||||
|
{"data": C(0x19191919, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0x09090909, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0x90909090, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0x985EFDB2, word_width), "k": Replicate(0, 4)},
|
||||||
|
]
|
||||||
|
yield from packet_sim(paks)
|
||||||
|
|
||||||
|
|
||||||
|
run_simulation(dut, testbench(), vcd_name="sim-cxp.vcd")
|
93
sim_buffer.py
Normal file
93
sim_buffer.py
Normal file
@ -0,0 +1,93 @@
|
|||||||
|
from migen import *
|
||||||
|
from misoc.interconnect.csr import *
|
||||||
|
from misoc.interconnect import stream
|
||||||
|
|
||||||
|
# from src.gateware.cxp_frame_pipeline import *
|
||||||
|
from src.gateware.cxp_pipeline import *
|
||||||
|
|
||||||
|
from types import SimpleNamespace
|
||||||
|
|
||||||
|
class DUT(Module):
|
||||||
|
def __init__(self):
|
||||||
|
# PHY
|
||||||
|
phy = SimpleNamespace()
|
||||||
|
phy.sink = stream.Endpoint(word_layout)
|
||||||
|
phy.source = stream.Endpoint(word_layout)
|
||||||
|
|
||||||
|
self.sync += [
|
||||||
|
phy.source.stb.eq(0),
|
||||||
|
If(~((phy.sink.data[:8] == 0xBC) & (phy.sink.k[0] == 1)),
|
||||||
|
phy.source.stb.eq(1),
|
||||||
|
phy.source.data.eq(phy.sink.data),
|
||||||
|
phy.source.k.eq(phy.sink.k),
|
||||||
|
),
|
||||||
|
]
|
||||||
|
|
||||||
|
# # #
|
||||||
|
|
||||||
|
dchar_decoder = Duplicated_Char_Decoder()
|
||||||
|
eop_marker = EOP_Marker()
|
||||||
|
|
||||||
|
pipeline = [phy, dchar_decoder, eop_marker]
|
||||||
|
self.submodules += pipeline[1:] #phy is not a submodules
|
||||||
|
|
||||||
|
for s, d in zip(pipeline, pipeline[1:]):
|
||||||
|
self.comb += s.source.connect(d.sink)
|
||||||
|
|
||||||
|
self.sink, self.source = pipeline[0].sink, pipeline[-1].source
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
dut = DUT()
|
||||||
|
|
||||||
|
|
||||||
|
def packet_sim(packets=[]):
|
||||||
|
print("=================TEST========================")
|
||||||
|
sink = dut.sink
|
||||||
|
source = dut.source
|
||||||
|
|
||||||
|
for i, p in enumerate(packets):
|
||||||
|
yield sink.data.eq(p["data"])
|
||||||
|
yield sink.k.eq(p["k"])
|
||||||
|
yield sink.stb.eq(p["stb"])
|
||||||
|
|
||||||
|
yield source.ack.eq(1)
|
||||||
|
# if i % 2 == 0:
|
||||||
|
# yield source.ack.eq(1)
|
||||||
|
# else:
|
||||||
|
# yield source.ack.eq(0)
|
||||||
|
|
||||||
|
yield
|
||||||
|
|
||||||
|
for _ in range(10):
|
||||||
|
yield sink.data.eq(0)
|
||||||
|
yield sink.k.eq(0)
|
||||||
|
yield sink.stb.eq(0)
|
||||||
|
yield sink.eop.eq(0)
|
||||||
|
|
||||||
|
yield source.ack.eq(1)
|
||||||
|
yield
|
||||||
|
|
||||||
|
assert True
|
||||||
|
|
||||||
|
|
||||||
|
def testbench():
|
||||||
|
paks = []
|
||||||
|
for i in range(1, 10):
|
||||||
|
paks += [
|
||||||
|
{"data": C(i << 8 | 1, word_width), "k" : Replicate(0, 4), "stb":1},
|
||||||
|
{"data": C(i << 8 | 2, word_width), "k" : Replicate(0, 4), "stb":1},
|
||||||
|
{"data": C(i << 8 | 3, word_width), "k" : Replicate(0, 4), "stb":1},
|
||||||
|
{"data": C(i << 8 | 4, word_width), "k" : Replicate(0, 4), "stb":1},
|
||||||
|
{"data": C(0xB53C3CBC, word_width), "k" : C(0b0111, 4), "stb":0},
|
||||||
|
{
|
||||||
|
"data": Replicate(KCode["pak_end"], 4),
|
||||||
|
"k": Replicate(1, 4),
|
||||||
|
"stb": 1,
|
||||||
|
},
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
yield from packet_sim(paks)
|
||||||
|
|
||||||
|
run_simulation(dut, testbench(), vcd_name="sim-cxp.vcd")
|
80
sim_double_buffer.py
Normal file
80
sim_double_buffer.py
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
from migen import *
|
||||||
|
from misoc.interconnect.csr import *
|
||||||
|
from misoc.interconnect import stream
|
||||||
|
|
||||||
|
from sim_generator import CXPCRC32Inserter
|
||||||
|
from src.gateware.cxp_frame_pipeline import *
|
||||||
|
from src.gateware.cxp_pipeline import *
|
||||||
|
|
||||||
|
class double_buffer_pipeline(Module):
|
||||||
|
def __init__(self):
|
||||||
|
fifo = stream.SyncFIFO(word_layout_dchar, 32)
|
||||||
|
double_buffer = CXPCRC32_Checker(0x100)
|
||||||
|
dchar_dropper = DChar_Dropper()
|
||||||
|
pipeline = [double_buffer, dchar_dropper]
|
||||||
|
self.submodules += pipeline
|
||||||
|
|
||||||
|
for s, d in zip(pipeline, pipeline[1:]):
|
||||||
|
self.comb += s.source.connect(d.sink)
|
||||||
|
|
||||||
|
self.sink = pipeline[0].sink
|
||||||
|
self.source = pipeline[-1].source
|
||||||
|
|
||||||
|
# for sim, no backpressure
|
||||||
|
self.comb += self.source.ack.eq(1)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
dut = double_buffer_pipeline()
|
||||||
|
|
||||||
|
|
||||||
|
def packet_sim(packets=[]):
|
||||||
|
print("=================TEST========================")
|
||||||
|
sink = dut.sink
|
||||||
|
|
||||||
|
cyc = len(packets)
|
||||||
|
pak = packets
|
||||||
|
for c in range(cyc):
|
||||||
|
yield sink.data.eq(pak[c]["data"])
|
||||||
|
yield sink.k.eq(pak[c]["k"])
|
||||||
|
yield sink.stb.eq(1)
|
||||||
|
if "eop" in pak[c]:
|
||||||
|
yield sink.eop.eq(1)
|
||||||
|
else:
|
||||||
|
yield sink.eop.eq(0)
|
||||||
|
yield
|
||||||
|
|
||||||
|
# extra clk cycles
|
||||||
|
for _ in range(cyc, cyc + 20):
|
||||||
|
yield sink.data.eq(0)
|
||||||
|
yield sink.k.eq(0)
|
||||||
|
yield sink.stb.eq(0)
|
||||||
|
yield sink.eop.eq(0)
|
||||||
|
yield
|
||||||
|
assert True
|
||||||
|
|
||||||
|
|
||||||
|
def testbench():
|
||||||
|
paks = [
|
||||||
|
{"data": C(0x7C7C7C7C, word_width), "k": Replicate(1, 4)},
|
||||||
|
{"data": C(0x01010101, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k": Replicate(0, 4)},
|
||||||
|
# {"data": C(0xF6ACEF6B, word_dw), "k": Replicate(0, 4), "eop":1},
|
||||||
|
{"data": C(0x6AEFACF6, word_width), "k": Replicate(0, 4), "eop":1},
|
||||||
|
|
||||||
|
|
||||||
|
{"data": C(0x19191919, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0x09090909, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0x90909090, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0xB2FD5E98, word_width), "k": Replicate(0, 4), "eop":1},
|
||||||
|
]
|
||||||
|
yield from packet_sim(paks)
|
||||||
|
|
||||||
|
|
||||||
|
run_simulation(dut, testbench(), vcd_name="sim-cxp.vcd")
|
72
sim_eop.py
Normal file
72
sim_eop.py
Normal file
@ -0,0 +1,72 @@
|
|||||||
|
from migen import *
|
||||||
|
from misoc.interconnect.csr import *
|
||||||
|
from misoc.interconnect import stream
|
||||||
|
|
||||||
|
from sim_generator import CXPCRC32Inserter
|
||||||
|
from src.gateware.cxp_frame_pipeline import *
|
||||||
|
from src.gateware.cxp_pipeline import *
|
||||||
|
|
||||||
|
class EOP_Pipeline(Module):
|
||||||
|
def __init__(self):
|
||||||
|
dchar_decoder = Duplicated_Char_Decoder()
|
||||||
|
eop_inserter = EOP_Marker()
|
||||||
|
buffer = stream.SyncFIFO(word_layout_dchar, 32)
|
||||||
|
pipeline = [dchar_decoder, eop_inserter, buffer]
|
||||||
|
self.submodules += pipeline
|
||||||
|
|
||||||
|
for s, d in zip(pipeline, pipeline[1:]):
|
||||||
|
self.comb += s.source.connect(d.sink)
|
||||||
|
|
||||||
|
self.sink = pipeline[0].sink
|
||||||
|
self.source = pipeline[-1].source
|
||||||
|
|
||||||
|
# for sim, no backpressure
|
||||||
|
self.comb += self.source.ack.eq(1)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
dut = EOP_Pipeline()
|
||||||
|
|
||||||
|
|
||||||
|
def packet_sim(packets=[]):
|
||||||
|
print("=================TEST========================")
|
||||||
|
sink = dut.sink
|
||||||
|
|
||||||
|
cyc = len(packets)
|
||||||
|
pak = packets
|
||||||
|
for c in range(cyc):
|
||||||
|
yield sink.data.eq(pak[c]["data"])
|
||||||
|
yield sink.k.eq(pak[c]["k"])
|
||||||
|
yield sink.stb.eq(1)
|
||||||
|
yield
|
||||||
|
|
||||||
|
# extra clk cycles
|
||||||
|
for _ in range(cyc, cyc + 10):
|
||||||
|
yield sink.data.eq(0)
|
||||||
|
yield sink.k.eq(0)
|
||||||
|
yield sink.stb.eq(0)
|
||||||
|
yield sink.eop.eq(0)
|
||||||
|
yield
|
||||||
|
assert True
|
||||||
|
|
||||||
|
|
||||||
|
def testbench():
|
||||||
|
paks = [
|
||||||
|
{"data": Replicate(KCode["pak_start"], 4), "k": Replicate(1, 4)},
|
||||||
|
{"data": C(0x1111, word_width), "k": Replicate(1, 4)},
|
||||||
|
{"data": C(0x2222, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0x3333, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0x4444, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(KCode["pak_end"], 4), "k": Replicate(1, 4)},
|
||||||
|
|
||||||
|
{"data": Replicate(KCode["pak_start"], 4), "k": Replicate(1, 4)},
|
||||||
|
{"data": C(0xAAAA, word_width), "k": Replicate(1, 4)},
|
||||||
|
{"data": C(0xBBBB, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0xCCCC, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": C(0xDDDD, word_width), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(KCode["pak_end"], 4), "k": Replicate(1, 4)},
|
||||||
|
]
|
||||||
|
yield from packet_sim(paks)
|
||||||
|
|
||||||
|
|
||||||
|
run_simulation(dut, testbench(), vcd_name="sim-cxp.vcd")
|
134
sim_frame.py
Normal file
134
sim_frame.py
Normal file
@ -0,0 +1,134 @@
|
|||||||
|
from migen import *
|
||||||
|
from misoc.interconnect import stream
|
||||||
|
|
||||||
|
from sim_generator import CXPCRC32Inserter
|
||||||
|
from sim_frame_gen import get_frame_packet
|
||||||
|
|
||||||
|
from src.gateware.cxp_pipeline import *
|
||||||
|
from src.gateware.cxp_frame_pipeline import *
|
||||||
|
|
||||||
|
import numpy as np
|
||||||
|
from PIL import Image
|
||||||
|
|
||||||
|
|
||||||
|
class Frame(Module):
|
||||||
|
def __init__(self):
|
||||||
|
# to construct correct crc and ack/stb signal
|
||||||
|
self.submodules.buffer = buffer = stream.SyncFIFO(word_layout, 32)
|
||||||
|
self.submodules.crc_inserter = crc_inserter = CXPCRC32Inserter()
|
||||||
|
self.submodules.dchar_decoder = dchar_decoder = Duplicated_Char_Decoder()
|
||||||
|
|
||||||
|
# NOTE: eop is needed for crc to work correctly and RX_Bootstrap need to be followed by a EOP marker anyway
|
||||||
|
self.submodules.eop_marker = eop_marker = EOP_Marker()
|
||||||
|
|
||||||
|
self.submodules.stream_pipe = stream_pipe = Pixel_Pipeline()
|
||||||
|
|
||||||
|
pipeline = [buffer, crc_inserter, dchar_decoder, eop_marker, stream_pipe]
|
||||||
|
for s, d in zip(pipeline, pipeline[1:]):
|
||||||
|
self.comb += s.source.connect(d.sink)
|
||||||
|
self.sink = pipeline[0].sink
|
||||||
|
self.source = pipeline[-1].source
|
||||||
|
|
||||||
|
# no backpressure for sim
|
||||||
|
self.sync += self.source.ack.eq(1)
|
||||||
|
|
||||||
|
|
||||||
|
dut = Frame()
|
||||||
|
|
||||||
|
|
||||||
|
def check_case(packet=[]):
|
||||||
|
print("=================TEST========================")
|
||||||
|
|
||||||
|
sink = dut.sink
|
||||||
|
stream_pipe = dut.stream_pipe
|
||||||
|
|
||||||
|
for i, p in enumerate(packet):
|
||||||
|
yield sink.data.eq(p["data"])
|
||||||
|
yield sink.k.eq(p["k"])
|
||||||
|
yield sink.stb.eq(1)
|
||||||
|
if "eop" in p:
|
||||||
|
yield sink.eop.eq(1)
|
||||||
|
else:
|
||||||
|
yield sink.eop.eq(0)
|
||||||
|
|
||||||
|
# check cycle result
|
||||||
|
yield
|
||||||
|
# source = dut.dchar_decoder.source
|
||||||
|
# source = dut.stream_pipe.frame_extractor.sink
|
||||||
|
source = dut.sink
|
||||||
|
# print(
|
||||||
|
# f"\nCYCLE#{i} : source char = {yield source.data:#X} k = {yield source.k:#X} stb = {yield source.stb} ack = {yield source.ack} eop = {yield source.eop}"
|
||||||
|
# )
|
||||||
|
|
||||||
|
# extra clk cycles
|
||||||
|
cyc = i + 1
|
||||||
|
img = []
|
||||||
|
line = -1
|
||||||
|
total_pixel = 1000
|
||||||
|
for i in range(cyc, cyc + total_pixel):
|
||||||
|
yield sink.data.eq(0)
|
||||||
|
yield sink.k.eq(0)
|
||||||
|
yield sink.stb.eq(0)
|
||||||
|
yield sink.eop.eq(0)
|
||||||
|
yield
|
||||||
|
# print(
|
||||||
|
# f"\nCYCLE#{i} : source char = {yield source.data:#X} k = {yield source.k:#X} stb = {yield source.stb} ack = {yield source.ack} eop = {yield source.eop}"
|
||||||
|
# )
|
||||||
|
|
||||||
|
frame_extractoer = dut.stream_pipe.frame_extractor
|
||||||
|
new_line = yield frame_extractoer.new_line
|
||||||
|
if new_line:
|
||||||
|
img.append([])
|
||||||
|
line += 1
|
||||||
|
|
||||||
|
stb = yield frame_extractoer.source.stb
|
||||||
|
data = yield frame_extractoer.source.data
|
||||||
|
if stb:
|
||||||
|
# CXP use MSB
|
||||||
|
img[line].append(np.uint16(data & 0xFFFF))
|
||||||
|
img[line].append(np.uint16(data >> 16))
|
||||||
|
|
||||||
|
# metadata = dut.stream_pipe.frame_extractor.metadata
|
||||||
|
# img_header_layout = [
|
||||||
|
# "stream_id",
|
||||||
|
# "source_tag",
|
||||||
|
# "x_size",
|
||||||
|
# "x_offset",
|
||||||
|
# "y_size",
|
||||||
|
# "y_offset",
|
||||||
|
# "l_size", # number of data words per image line
|
||||||
|
# "pixel_format",
|
||||||
|
# "tap_geo",
|
||||||
|
# "flag",
|
||||||
|
# ]
|
||||||
|
# for name in img_header_layout:
|
||||||
|
# print(f"{name} = {yield getattr(metadata, name):#04X} ", end="")
|
||||||
|
# print()
|
||||||
|
Image.fromarray(np.array(img, dtype=np.uint8)).show()
|
||||||
|
assert True
|
||||||
|
|
||||||
|
|
||||||
|
def testbench():
|
||||||
|
stream_id = 0x69
|
||||||
|
packet_tag = 0
|
||||||
|
frame_packet = get_frame_packet(stream_id)
|
||||||
|
packet = [
|
||||||
|
{"data": Replicate(C(stream_id, char_width), 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(C(packet_tag, char_width), 4), "k": Replicate(0, 4)},
|
||||||
|
{
|
||||||
|
"data": Replicate(C(len(frame_packet), 2*char_width)[8:], 4),
|
||||||
|
"k": Replicate(0, 4),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"data": Replicate(C(len(frame_packet), 2*char_width)[:8], 4),
|
||||||
|
"k": Replicate(0, 4),
|
||||||
|
},
|
||||||
|
]
|
||||||
|
packet += frame_packet
|
||||||
|
# NOTE: for crc inserter!!!!
|
||||||
|
packet[-1]["eop"] = 0
|
||||||
|
|
||||||
|
yield from check_case(packet)
|
||||||
|
|
||||||
|
|
||||||
|
run_simulation(dut, testbench(), vcd_name="sim-cxp.vcd")
|
148
sim_frame_gen.py
Normal file
148
sim_frame_gen.py
Normal file
@ -0,0 +1,148 @@
|
|||||||
|
from migen import *
|
||||||
|
from misoc.interconnect import stream
|
||||||
|
|
||||||
|
from src.gateware.cxp_pipeline import *
|
||||||
|
from src.gateware.cxp_frame_pipeline import *
|
||||||
|
|
||||||
|
from PIL import Image
|
||||||
|
import numpy as np
|
||||||
|
|
||||||
|
|
||||||
|
def get_image_header(
|
||||||
|
stream_id, source_tag, xsize, xoffset, ysize, yoffset, dsize, pixelF, tag_geo, flag
|
||||||
|
):
|
||||||
|
stream_id = C(stream_id, char_width)
|
||||||
|
source_tag = C(source_tag, 2 * char_width)
|
||||||
|
xsize = C(xsize, 3 * char_width)
|
||||||
|
xoffset = C(xoffset, 3 * char_width)
|
||||||
|
ysize = C(ysize, 3 * char_width)
|
||||||
|
yoffset = C(yoffset, 3 * char_width)
|
||||||
|
dsize = C(dsize, 3 * char_width)
|
||||||
|
pixelF = C(pixelF, 2 * char_width)
|
||||||
|
tag_geo = C(tag_geo, 2 * char_width)
|
||||||
|
flag = C(flag, char_width)
|
||||||
|
|
||||||
|
assert len(stream_id) == len(flag) == char_width
|
||||||
|
assert len(source_tag) == len(pixelF) == len(tag_geo) == 2 * char_width
|
||||||
|
assert len(xsize) == len(xoffset) == len(ysize) == len(yoffset) == 3 * char_width
|
||||||
|
return [
|
||||||
|
{"data": Replicate(KCode["stream_marker"], 4), "k": Replicate(1, 4)},
|
||||||
|
{"data": Replicate(C(0x01, char_width), 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(stream_id, 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(source_tag[8:], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(source_tag[:8], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(xsize[16:], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(xsize[8:16], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(xsize[:8], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(xoffset[16:], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(xoffset[8:16], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(xoffset[:8], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(ysize[16:], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(ysize[8:16], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(ysize[:8], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(yoffset[16:], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(yoffset[8:16], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(yoffset[:8], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(dsize[16:], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(dsize[8:16], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(dsize[:8], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(pixelF[8:], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(pixelF[:8], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(tag_geo[8:], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(tag_geo[:8], 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(flag, 4), "k": Replicate(0, 4)},
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
def get_line_marker():
|
||||||
|
return [
|
||||||
|
{"data": Replicate(KCode["stream_marker"], 4), "k": Replicate(1, 4)},
|
||||||
|
{"data": Replicate(C(0x02, char_width), 4), "k": Replicate(0, 4)},
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
def get_frame_packet(stream_id, pixel_format="mono16"):
|
||||||
|
assert pixel_format in ["mono16"]
|
||||||
|
|
||||||
|
arr = [
|
||||||
|
[204, 200, 203, 205, 190, 187, 189, 205, 214, 197, 188, 185, 181, 178, 193, 209, 211, 207, 211, 192, 168, 168, 171, 199, 210, 212, 203, 196],
|
||||||
|
[218, 205, 199, 190, 192, 197, 196, 195, 184, 178, 182, 173, 166, 132, 122, 114, 154, 184, 187, 188, 171, 168, 170, 180, 192, 196, 202, 198],
|
||||||
|
[223, 222, 222, 224, 216, 199, 199, 207, 205, 189, 183, 182, 144, 66, 61, 66, 80, 148, 181, 175, 169, 170, 174, 177, 196, 206, 223, 218],
|
||||||
|
[221, 226, 225, 222, 211, 200, 202, 208, 215, 201, 187, 180, 133, 116, 113, 118, 96, 111, 206, 193, 170, 169, 186, 211, 218, 224, 231, 223],
|
||||||
|
[219, 216, 206, 197, 210, 201, 206, 203, 191, 190, 185, 145, 134, 140, 159, 170, 150, 116, 180, 173, 179, 170, 172, 185, 201, 218, 227, 227],
|
||||||
|
[203, 198, 194, 208, 227, 201, 201, 201, 215, 221, 209, 170, 136, 113, 141, 139, 141, 145, 188, 170, 180, 169, 184, 173, 174, 192, 215, 230],
|
||||||
|
[206, 224, 213, 213, 233, 207, 204, 226, 233, 227, 214, 166, 145, 123, 145, 155, 147, 186, 213, 187, 171, 169, 193, 193, 171, 178, 186, 207],
|
||||||
|
[212, 228, 216, 205, 214, 205, 204, 230, 235, 225, 219, 187, 143, 122, 146, 163, 158, 195, 209, 203, 174, 168, 190, 185, 187, 202, 180, 174],
|
||||||
|
[197, 206, 201, 223, 213, 201, 203, 231, 234, 225, 218, 206, 147, 125, 149, 155, 190, 208, 206, 203, 175, 168, 171, 179, 184, 206, 189, 176],
|
||||||
|
[213, 202, 209, 235, 223, 200, 202, 202, 227, 227, 202, 176, 138, 122, 144, 153, 190, 209, 207, 191, 172, 167, 179, 204, 190, 191, 180, 193],
|
||||||
|
[225, 225, 207, 231, 219, 197, 215, 200, 194, 199, 181, 172, 131, 129, 147, 159, 113, 175, 196, 179, 184, 169, 181, 210, 202, 204, 200, 177],
|
||||||
|
[208, 222, 204, 223, 210, 191, 195, 198, 203, 167, 171, 168, 135, 129, 149, 175, 66, 57, 90, 121, 147, 165, 181, 205, 195, 217, 209, 173],
|
||||||
|
[188, 216, 201, 206, 199, 180, 185, 180, 129, 75, 139, 166, 124, 146, 189, 135, 51, 41, 38, 40, 45, 63, 131, 201, 189, 215, 193, 170],
|
||||||
|
[188, 194, 195, 192, 182, 180, 134, 68, 45, 41, 96, 130, 116, 156, 163, 64, 46, 41, 43, 41, 42, 42, 74, 181, 177, 198, 175, 193],
|
||||||
|
[179, 179, 209, 224, 198, 182, 99, 42, 44, 41, 44, 100, 116, 125, 100, 46, 45, 42, 42, 37, 44, 43, 49, 150, 183, 170, 172, 198],
|
||||||
|
[175, 177, 208, 223, 197, 180, 94, 40, 42, 40, 41, 99, 134, 117, 80, 43, 46, 43, 37, 37, 44, 42, 35, 129, 195, 170, 170, 180],
|
||||||
|
[179, 181, 187, 217, 193, 175, 91, 38, 41, 41, 42, 106, 151, 107, 62, 43, 45, 41, 33, 38, 42, 34, 33, 77, 188, 175, 173, 208],
|
||||||
|
[190, 191, 180, 213, 194, 175, 78, 38, 40, 40, 40, 98, 134, 97, 51, 44, 59, 50, 37, 40, 36, 26, 36, 44, 100, 178, 192, 206],
|
||||||
|
[199, 191, 184, 204, 196, 176, 78, 33, 38, 38, 39, 80, 102, 83, 43, 44, 112, 130, 122, 63, 33, 24, 29, 34, 33, 74, 162, 195],
|
||||||
|
[191, 170, 196, 193, 186, 177, 88, 27, 34, 37, 36, 74, 101, 70, 36, 37, 81, 127, 137, 113, 40, 28, 30, 32, 36, 29, 69, 173],
|
||||||
|
[164, 189, 190, 180, 176, 172, 83, 26, 28, 33, 32, 68, 97, 62, 32, 30, 44, 97, 123, 136, 58, 42, 44, 43, 43, 40, 58, 162],
|
||||||
|
[177, 202, 205, 181, 174, 163, 78, 38, 35, 47, 54, 67, 92, 51, 28, 29, 26, 21, 39, 85, 47, 46, 52, 47, 46, 45, 48, 141],
|
||||||
|
[181, 193, 199, 192, 171, 163, 91, 67, 121, 123, 91, 63, 89, 45, 25, 25, 23, 20, 15, 13, 20, 48, 54, 35, 34, 34, 68, 146],
|
||||||
|
[175, 192, 195, 179, 165, 163, 100, 64, 99, 94, 82, 58, 83, 37, 23, 22, 22, 27, 21, 15, 14, 44, 98, 83, 94, 118, 164, 157],
|
||||||
|
[153, 184, 171, 163, 161, 157, 140, 70, 58, 89, 61, 53, 76, 30, 20, 20, 20, 31, 24, 19, 16, 47, 159, 163, 160, 171, 160, 142],
|
||||||
|
[142, 150, 161, 168, 154, 154, 164, 138, 76, 55, 26, 37, 62, 24, 19, 19, 20, 21, 23, 27, 31, 46, 142, 156, 151, 153, 147, 145],
|
||||||
|
[153, 147, 174, 171, 151, 150, 169, 158, 142, 92, 28, 60, 59, 20, 20, 18, 20, 26, 27, 29, 33, 38, 125, 153, 150, 147, 147, 148],
|
||||||
|
[138, 141, 166, 164, 146, 144, 164, 149, 132, 72, 34, 88, 72, 24, 19, 18, 18, 23, 25, 28, 31, 30, 98, 150, 146, 144, 146, 144]
|
||||||
|
]
|
||||||
|
source_tag = 0
|
||||||
|
xsize, ysize = len(arr[0]), len(arr)
|
||||||
|
|
||||||
|
xoffset, yoffset = 0, 0
|
||||||
|
if pixel_format == "mono16":
|
||||||
|
dsize = xsize // 2
|
||||||
|
pixelF = 0x0105
|
||||||
|
tag_geo = 0
|
||||||
|
flag = 0
|
||||||
|
|
||||||
|
packet = []
|
||||||
|
# Image header
|
||||||
|
packet += get_image_header(
|
||||||
|
stream_id,
|
||||||
|
source_tag,
|
||||||
|
xsize,
|
||||||
|
xoffset,
|
||||||
|
ysize,
|
||||||
|
yoffset,
|
||||||
|
dsize,
|
||||||
|
pixelF,
|
||||||
|
tag_geo,
|
||||||
|
flag,
|
||||||
|
)
|
||||||
|
|
||||||
|
for line in arr:
|
||||||
|
packet += get_line_marker()
|
||||||
|
if pixel_format == "mono16":
|
||||||
|
for i in range(len(line)):
|
||||||
|
if (i % 2) == 0:
|
||||||
|
if i == len(line) - 1:
|
||||||
|
# print(C(line[i]))
|
||||||
|
packet += [
|
||||||
|
{
|
||||||
|
"data": C(line[i], 4 * char_width),
|
||||||
|
"k": Replicate(0, 4),
|
||||||
|
},
|
||||||
|
]
|
||||||
|
else:
|
||||||
|
# print(C(line[i], 2 * char_width), C(line[i + 1]))
|
||||||
|
# CXP use MSB
|
||||||
|
packet += [
|
||||||
|
{
|
||||||
|
"data": Cat(
|
||||||
|
C(line[i], 2 * char_width),
|
||||||
|
C(line[i + 1], 2 * char_width),
|
||||||
|
),
|
||||||
|
"k": Replicate(0, 4),
|
||||||
|
},
|
||||||
|
]
|
||||||
|
|
||||||
|
return packet
|
122
sim_generator.py
Normal file
122
sim_generator.py
Normal file
@ -0,0 +1,122 @@
|
|||||||
|
from migen import *
|
||||||
|
from misoc.interconnect.csr import *
|
||||||
|
from misoc.interconnect import stream
|
||||||
|
|
||||||
|
from src.gateware.cxp_frame_pipeline import CXPCRC32
|
||||||
|
from src.gateware.cxp_pipeline import *
|
||||||
|
|
||||||
|
class CXPCRC32Inserter(Module):
|
||||||
|
def __init__(self, insert_eop=True):
|
||||||
|
self.sink = stream.Endpoint(word_layout)
|
||||||
|
self.source = stream.Endpoint(word_layout)
|
||||||
|
|
||||||
|
# # #
|
||||||
|
|
||||||
|
self.submodules.crc = crc = CXPCRC32(word_width)
|
||||||
|
self.submodules.fsm = fsm = FSM(reset_state="IDLE")
|
||||||
|
|
||||||
|
# WARNING: this will eat data if the source don't care about ack
|
||||||
|
# NOTE: need one cycle to turn itself on
|
||||||
|
fsm.act(
|
||||||
|
"IDLE",
|
||||||
|
crc.reset.eq(1),
|
||||||
|
self.sink.ack.eq(1),
|
||||||
|
If(
|
||||||
|
self.sink.stb,
|
||||||
|
self.sink.ack.eq(0),
|
||||||
|
NextState("COPY"),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
fsm.act(
|
||||||
|
"COPY",
|
||||||
|
crc.ce.eq(self.sink.stb & self.source.ack),
|
||||||
|
crc.data.eq(self.sink.data),
|
||||||
|
self.sink.connect(self.source),
|
||||||
|
self.source.eop.eq(0),
|
||||||
|
If(
|
||||||
|
self.sink.stb & self.sink.eop & self.source.ack,
|
||||||
|
NextState("INSERT"),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
fsm.act(
|
||||||
|
"INSERT",
|
||||||
|
self.source.stb.eq(1),
|
||||||
|
self.source.eop.eq(1) if insert_eop else self.source.eop.eq(0),
|
||||||
|
self.source.data.eq(crc.value),
|
||||||
|
If(self.source.ack, NextState("IDLE")),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class StreamPacket_Wrapper(Module):
|
||||||
|
def __init__(self):
|
||||||
|
self.sink = stream.Endpoint(word_layout)
|
||||||
|
self.source = stream.Endpoint(word_layout)
|
||||||
|
|
||||||
|
# # #
|
||||||
|
|
||||||
|
self.submodules.fsm = fsm = FSM(reset_state="IDLE")
|
||||||
|
|
||||||
|
fsm.act(
|
||||||
|
"IDLE",
|
||||||
|
self.sink.ack.eq(1),
|
||||||
|
If(
|
||||||
|
self.sink.stb,
|
||||||
|
self.sink.ack.eq(0),
|
||||||
|
NextState("INSERT_HEADER_0"),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
fsm.act(
|
||||||
|
"INSERT_HEADER_0",
|
||||||
|
self.sink.ack.eq(0),
|
||||||
|
self.source.stb.eq(1),
|
||||||
|
self.source.data.eq(Replicate(KCode["pak_start"], 4)),
|
||||||
|
self.source.k.eq(Replicate(1, 4)),
|
||||||
|
If(self.source.ack, NextState("INSERT_HEADER_1")),
|
||||||
|
)
|
||||||
|
|
||||||
|
fsm.act(
|
||||||
|
"INSERT_HEADER_1",
|
||||||
|
self.sink.ack.eq(0),
|
||||||
|
self.source.stb.eq(1),
|
||||||
|
self.source.data.eq(Replicate(C(0x01, char_width), 4)),
|
||||||
|
self.source.k.eq(Replicate(0, 4)),
|
||||||
|
If(self.source.ack, NextState("COPY")),
|
||||||
|
)
|
||||||
|
|
||||||
|
fsm.act(
|
||||||
|
"COPY",
|
||||||
|
self.sink.connect(self.source),
|
||||||
|
self.source.eop.eq(0),
|
||||||
|
If(
|
||||||
|
self.sink.stb & self.sink.eop & self.source.ack,
|
||||||
|
NextState("INSERT_FOOTER"),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
fsm.act(
|
||||||
|
"INSERT_FOOTER",
|
||||||
|
self.sink.ack.eq(0),
|
||||||
|
self.source.stb.eq(1),
|
||||||
|
self.source.data.eq(Replicate(KCode["pak_end"], 4)),
|
||||||
|
self.source.k.eq(Replicate(1, 4)),
|
||||||
|
# Simulate RX don't have eop tagged
|
||||||
|
# self.source.eop.eq(1),
|
||||||
|
If(self.source.ack, NextState("IDLE")),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
# With KCode & 0x01*4
|
||||||
|
class StreamData_Generator(Module):
|
||||||
|
def __init__(self):
|
||||||
|
# should be big enough for all test
|
||||||
|
self.submodules.buffer = buffer = stream.SyncFIFO(word_layout, 32)
|
||||||
|
self.submodules.crc_inserter = crc_inserter = CXPCRC32Inserter()
|
||||||
|
self.submodules.wrapper = wrapper = StreamPacket_Wrapper()
|
||||||
|
|
||||||
|
# # #
|
||||||
|
pipeline = [buffer, crc_inserter, wrapper]
|
||||||
|
for s, d in zip(pipeline, pipeline[1:]):
|
||||||
|
self.comb += s.source.connect(d.sink)
|
||||||
|
self.sink = pipeline[0].sink
|
||||||
|
self.source = pipeline[-1].source
|
323
sim_roi.py
Normal file
323
sim_roi.py
Normal file
@ -0,0 +1,323 @@
|
|||||||
|
from migen import *
|
||||||
|
from misoc.interconnect.csr import *
|
||||||
|
from misoc.interconnect import stream
|
||||||
|
|
||||||
|
from src.gateware.cxp_frame_pipeline import *
|
||||||
|
from src.gateware.cxp_pipeline import *
|
||||||
|
|
||||||
|
from math import ceil
|
||||||
|
|
||||||
|
class ROI(Module):
|
||||||
|
def __init__(self):
|
||||||
|
fifo = stream.SyncFIFO(word_layout, 32) # to avoid data getting eaten and act as delay between eop
|
||||||
|
dchar_decoder = Duplicated_Char_Decoder()
|
||||||
|
# self.crc = CXPCRC32_Checker()
|
||||||
|
self.pipeline = Pixel_Pipeline(24, 32)
|
||||||
|
|
||||||
|
pipeline = [fifo, dchar_decoder, self.pipeline]
|
||||||
|
self.submodules += pipeline
|
||||||
|
|
||||||
|
for s, d in zip(pipeline, pipeline[1:]):
|
||||||
|
self.comb += s.source.connect(d.sink)
|
||||||
|
|
||||||
|
self.sink = pipeline[0].sink
|
||||||
|
# self.source = pipeline[-1].source
|
||||||
|
|
||||||
|
# DEBUG: test roi
|
||||||
|
cfg = self.pipeline.roi.cfg
|
||||||
|
self.comb += [
|
||||||
|
cfg.x0.eq(1),
|
||||||
|
cfg.x1.eq(2),
|
||||||
|
cfg.y0.eq(1),
|
||||||
|
cfg.y1.eq(2),
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
dut = ROI()
|
||||||
|
|
||||||
|
|
||||||
|
def packet_sim(packets=[]):
|
||||||
|
print("=================TEST========================")
|
||||||
|
sink = dut.sink
|
||||||
|
|
||||||
|
cyc = len(packets)
|
||||||
|
pak = packets
|
||||||
|
for c in range(cyc):
|
||||||
|
yield sink.data.eq(pak[c]["data"])
|
||||||
|
yield sink.k.eq(pak[c]["k"])
|
||||||
|
yield sink.stb.eq(1)
|
||||||
|
if "eop" in pak[c]:
|
||||||
|
yield sink.eop.eq(1)
|
||||||
|
else:
|
||||||
|
yield sink.eop.eq(0)
|
||||||
|
yield
|
||||||
|
|
||||||
|
# extra clk cycles
|
||||||
|
for _ in range(cyc, cyc + 20):
|
||||||
|
yield sink.data.eq(0)
|
||||||
|
yield sink.k.eq(0)
|
||||||
|
yield sink.stb.eq(0)
|
||||||
|
yield sink.eop.eq(0)
|
||||||
|
yield
|
||||||
|
metadata = dut.pipeline.header_decoder.metadata
|
||||||
|
img_header_layout = [
|
||||||
|
"stream_id",
|
||||||
|
"source_tag", # image index since powering on
|
||||||
|
"x_size",
|
||||||
|
"x_offset",
|
||||||
|
"y_size",
|
||||||
|
"y_offset",
|
||||||
|
"l_size", # number of data words per image line
|
||||||
|
"pixel_format",
|
||||||
|
"tap_geo",
|
||||||
|
"flag",
|
||||||
|
]
|
||||||
|
for name in img_header_layout:
|
||||||
|
print(f"{name} = {yield getattr(metadata, name):#04X} ", end="")
|
||||||
|
print()
|
||||||
|
|
||||||
|
roi = dut.pipeline.roi
|
||||||
|
print(f"x0 = {yield roi.cfg.x0} y0 = {yield roi.cfg.y0} | x1 = {yield roi.cfg.x1} y1 = {yield roi.cfg.y1}")
|
||||||
|
print(f"out count = {yield roi.out.count}")
|
||||||
|
assert True
|
||||||
|
|
||||||
|
|
||||||
|
def testbench():
|
||||||
|
paks = [
|
||||||
|
{"data": C(0x7C7C7C7C, word_width), "k" : Replicate(1, 4)},
|
||||||
|
{"data": C(0x01010101, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)}, # stream id
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x6AEFACF6, word_width), "k" : Replicate(0, 4), "eop":0}, # crc
|
||||||
|
|
||||||
|
{"data": C(0x02020202, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)}, # Xsize[23:16]
|
||||||
|
{"data": C(0x09090909, word_width), "k" : Replicate(0, 4)}, # Xsize[15:8]
|
||||||
|
{"data": C(0x90909090, word_width), "k" : Replicate(0, 4)}, # Xsize[7:0]
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x8EE1DAA1, word_width), "k" : Replicate(0, 4), "eop":0}, # crc
|
||||||
|
|
||||||
|
{"data": C(0x08080808, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x08080808, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)}, # DsizeL[23:16]
|
||||||
|
{"data": C(0x02020202, word_width), "k" : Replicate(0, 4)}, # DsizeL[15:8]
|
||||||
|
{"data": C(0x64646464, word_width), "k" : Replicate(0, 4)}, # DsizeL[7:0]
|
||||||
|
{"data": C(0x01010101, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x01010101, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x51C243EA, word_width), "k" : Replicate(0, 4), "eop":0}, # crc
|
||||||
|
|
||||||
|
# the new line + pixel data
|
||||||
|
{"data": C(0x7C7C7C7C, word_width), "k" : Replicate(1, 4)},
|
||||||
|
{"data": C(0x02020202, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0D0D0C0C, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0B0C0C, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0C0B0D, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0D0B0C0B, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0C0C0C0C, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0B0C0B, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0B0B0B, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0C0B0B0C, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0C0B0C0B, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0B0A0B, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0C0B0B, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0C0C0C0B, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0C0B0C, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0C0C0B0C, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0C0B0B0B, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0C0C0C0B, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0C0B0C0B, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0C0C0C0C, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0C0B0B, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0C0B0B0B, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0C0B0C0B, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0B0D0A, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0C0C0C, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0B0B0C, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0B0C0C, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0B0C0B, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0C0C0C, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0C0C0C, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0A0B0D, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0B0C0C, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0C0B0C, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0D0C0C0B, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0A0C0B, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0D0D0B0A, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0B0C0C, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0C0C0B, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0C0B0B0B, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0C0C0B, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0B0B0C, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x0B0B0B0C, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0xCB5DCDD6, word_width), "k" : Replicate(0, 4), "eop": 0}, # crc
|
||||||
|
|
||||||
|
# {"data": C(0x0C0B0C0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0D0B0B0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0D0B0C0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0C0C0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0B0C0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0C0B0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0B0C0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0D0B0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0C0C0A, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0D0C0B0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0D0C0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0C0C0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0C0B0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0C0C0D, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0A0C0C0D, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0B0B0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0B0C0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0C0B0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0B0B0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0B0B0A, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0C0C0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0B0B0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0C0C0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0C0B0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0C0B0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0C0C0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0C0B0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0C0D0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0C0C0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0B0B0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0C0B0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0B0B0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0B0C0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0B0B0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0B0C0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0D0B0D0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0B0D0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0B0C0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0B0B0A, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0C0C0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0C0B0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0B0A0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0C0C0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0D0C0C0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0C0B0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0C0C0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0C0C0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0B0B0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0B0B0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0B0C0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0C0C0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0C0D0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0D0C0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0C0D0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0C0D0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0B0C0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0C0B0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0C0D0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0C0B0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0D0C0D0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0C0C0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0D0B0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0D0C0C0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0B0C0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0D0D0B0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0C0B0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0D0B0C0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0C0B0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0C0B0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0C0C0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0C0B0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0B0C0B, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0B0C0C0D, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0D0C0C0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0C0B0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0C0C0C0C, word_dw), "k" : Replicate(0, 4)},
|
||||||
|
# {"data": C(0x0BADF020, word_dw), "k" : Replicate(0, 4), "eop":0}, # crc
|
||||||
|
]
|
||||||
|
yield from packet_sim(paks)
|
||||||
|
|
||||||
|
|
||||||
|
def testbench_fake_data():
|
||||||
|
# config
|
||||||
|
pix_size = 8
|
||||||
|
# x_size = 2448
|
||||||
|
x_size = 10
|
||||||
|
y_size = 2
|
||||||
|
|
||||||
|
|
||||||
|
l_size = ceil(x_size*pix_size/32)
|
||||||
|
pix_fmt = {
|
||||||
|
8: 0x0101,
|
||||||
|
10: 0x0102,
|
||||||
|
12: 0x0103,
|
||||||
|
14: 0x0104,
|
||||||
|
16: 0x0105,
|
||||||
|
}
|
||||||
|
# frame header
|
||||||
|
paks = [
|
||||||
|
{"data": C(0x7C7C7C7C, word_width), "k" : Replicate(1, 4)},
|
||||||
|
{"data": C(0x01010101, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)}, # stream id
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x6AEFACF6, word_width), "k" : Replicate(0, 4), "eop":0}, # fake crc
|
||||||
|
|
||||||
|
{"data": C(0x02020202, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)}, # Xsize[23:16]
|
||||||
|
{"data": Replicate(C(x_size >> 8, char_width), 4), "k" : Replicate(0, 4)}, # Xsize[15:8]
|
||||||
|
{"data": Replicate(C(x_size & 0xFF, char_width), 4), "k" : Replicate(0, 4)}, # Xsize[7:0]
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)}, # Ysize[23:16]
|
||||||
|
{"data": C(0x8EE1DAA1, word_width), "k" : Replicate(0, 4), "eop":0}, # fake crc
|
||||||
|
|
||||||
|
{"data": Replicate(C(y_size >> 8, char_width), 4), "k" : Replicate(0, 4)}, # Ysize[15:8]
|
||||||
|
{"data": Replicate(C(y_size & 0xFF, char_width), 4), "k" : Replicate(0, 4)}, # Ysize[7:0]
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x08080808, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)}, # DsizeL[23:16]
|
||||||
|
{"data": Replicate(C(l_size >> 8, char_width), 4), "k" : Replicate(0, 4)}, # DsizeL[15:8]
|
||||||
|
{"data": Replicate(C(l_size & 0xFF, char_width), 4), "k" : Replicate(0, 4)}, # DsizeL[7:0]
|
||||||
|
{"data": Replicate(C(pix_fmt[pix_size] >> 8, char_width), 4), "k" : Replicate(0, 4)}, # PixelF[15:8]
|
||||||
|
{"data": Replicate(C(pix_fmt[pix_size] & 0xFF, char_width), 4), "k" : Replicate(0, 4)}, # PixelF[7:0]
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x00000000, word_width), "k" : Replicate(0, 4)},
|
||||||
|
{"data": C(0x51C243EA, word_width), "k" : Replicate(0, 4), "eop":0}, # fake crc
|
||||||
|
]
|
||||||
|
|
||||||
|
# pixel data
|
||||||
|
base = 0
|
||||||
|
for _ in range(y_size):
|
||||||
|
pix = []
|
||||||
|
for _ in range(x_size):
|
||||||
|
base += 1
|
||||||
|
pix.append(base)
|
||||||
|
print(pix)
|
||||||
|
|
||||||
|
packed = 0
|
||||||
|
for i, p in enumerate(pix):
|
||||||
|
packed += p << i*pix_size
|
||||||
|
# print(f"{packed:08X}")
|
||||||
|
|
||||||
|
# new line indicator
|
||||||
|
paks += [
|
||||||
|
{"data": C(0x7C7C7C7C, word_width), "k" : Replicate(1, 4)},
|
||||||
|
{"data": C(0x02020202, word_width), "k" : Replicate(0, 4)},
|
||||||
|
]
|
||||||
|
for i in range(l_size):
|
||||||
|
serialized = (packed & (0xFFFF_FFFF << i*word_width)) >> i*word_width
|
||||||
|
print(f"{serialized:08X}")
|
||||||
|
paks.append({"data": C(serialized, word_width), "k" : Replicate(0, 4)})
|
||||||
|
|
||||||
|
paks.append({"data": C(0xCB5DCDD6, word_width), "k" : Replicate(0, 4), "eop": 0}) # fake crc
|
||||||
|
|
||||||
|
yield from packet_sim(paks)
|
||||||
|
|
||||||
|
run_simulation(dut, testbench_fake_data(), vcd_name="sim-cxp.vcd")
|
154
sim_stream.py
Normal file
154
sim_stream.py
Normal file
@ -0,0 +1,154 @@
|
|||||||
|
from migen import *
|
||||||
|
from misoc.interconnect import stream
|
||||||
|
|
||||||
|
from sim_pipeline import *
|
||||||
|
from sim_generator import StreamData_Generator
|
||||||
|
|
||||||
|
from src.gateware.cxp_pipeline import *
|
||||||
|
|
||||||
|
class CXP_Links(Module):
|
||||||
|
def __init__(self):
|
||||||
|
# TODO: select the correct buffer to read from
|
||||||
|
# NOTE: although there are double buffer in each connect, the reading must be faster than writing to avoid data loss
|
||||||
|
|
||||||
|
self.downconn_sources = []
|
||||||
|
self.stream_sinks = []
|
||||||
|
for i in range(2):
|
||||||
|
downconn = Pipeline()
|
||||||
|
setattr(self.submodules, "cxp_conn"+str(i), downconn)
|
||||||
|
self.downconn_sources.append(downconn)
|
||||||
|
|
||||||
|
stream_pipeline = Stream_Pipeline()
|
||||||
|
setattr(self.submodules, "stream_pipeline"+str(i), stream_pipeline)
|
||||||
|
self.stream_sinks.append(stream_pipeline)
|
||||||
|
|
||||||
|
self.submodules.crossbar = Streams_Crossbar(self.downconn_sources, self.stream_sinks)
|
||||||
|
|
||||||
|
class Pipeline(Module):
|
||||||
|
def __init__(self):
|
||||||
|
self.submodules.generator = generator = StreamData_Generator()
|
||||||
|
self.submodules.dchar_decoder = dchar_decoder = Duplicated_Char_Decoder()
|
||||||
|
self.submodules.data_decoder = data_decoder = Control_Packet_Reader()
|
||||||
|
self.submodules.eop_marker = eop_marker = EOP_Marker()
|
||||||
|
|
||||||
|
# # #
|
||||||
|
|
||||||
|
pipeline = [generator, dchar_decoder, data_decoder, eop_marker]
|
||||||
|
for s, d in zip(pipeline, pipeline[1:]):
|
||||||
|
self.comb += s.source.connect(d.sink)
|
||||||
|
self.sink = pipeline[0].sink
|
||||||
|
self.source = pipeline[-1].source
|
||||||
|
|
||||||
|
# self.comb += self.source.ack.eq(1)
|
||||||
|
|
||||||
|
|
||||||
|
dut = CXP_Links()
|
||||||
|
|
||||||
|
def check_case(packet=[]):
|
||||||
|
print("=================TEST========================")
|
||||||
|
|
||||||
|
downconns = dut.downconn_sources
|
||||||
|
stream_buffers = dut.stream_sinks
|
||||||
|
ch = 0
|
||||||
|
|
||||||
|
for i, p in enumerate(packet):
|
||||||
|
for x in range(len(downconns)):
|
||||||
|
if x == ch:
|
||||||
|
yield downconns[x].sink.data.eq(p["data"])
|
||||||
|
yield downconns[x].sink.k.eq(p["k"])
|
||||||
|
yield downconns[x].sink.stb.eq(1)
|
||||||
|
else:
|
||||||
|
yield downconns[x].sink.data.eq(0)
|
||||||
|
yield downconns[x].sink.k.eq(0)
|
||||||
|
yield downconns[x].sink.stb.eq(0)
|
||||||
|
yield downconns[x].sink.eop.eq(0)
|
||||||
|
if "eop" in p:
|
||||||
|
yield downconns[ch].sink.eop.eq(1)
|
||||||
|
# compensate for delay
|
||||||
|
# yield
|
||||||
|
# yield downconns[ch].sink.data.eq(0)
|
||||||
|
# yield downconns[ch].sink.k.eq(0)
|
||||||
|
# yield downconns[ch].sink.stb.eq(0)
|
||||||
|
# yield downconns[ch].sink.eop.eq(0)
|
||||||
|
# yield
|
||||||
|
# yield
|
||||||
|
# yield
|
||||||
|
ch = (ch + 1) % len(downconns)
|
||||||
|
else:
|
||||||
|
yield downconns[ch].sink.eop.eq(0)
|
||||||
|
|
||||||
|
# check cycle result
|
||||||
|
yield
|
||||||
|
# source = dut.stream_pipeline_sinks[0].source
|
||||||
|
source = dut.stream_sinks[0].double_buffer.source
|
||||||
|
print(
|
||||||
|
f"\nCYCLE#{i} : source char = {yield source.data:#X} k = {yield source.k:#X} stb = {yield source.stb} ack = {yield source.ack} eop = {yield source.eop}"
|
||||||
|
# f" source dchar = {yield source.dchar:#X} dchar_k = {yield source.dchar_k:#X}"
|
||||||
|
f"\nCYCLE#{i} : read mask = {yield dut.crossbar.mux.sel}"
|
||||||
|
# f"\nCYCLE#{i} : stream id = {yield decoder.stream_id:#X} pak_tag = {yield decoder.pak_tag:#X}"
|
||||||
|
# f" stream_pak_size = {yield decoder.stream_pak_size:#X}"
|
||||||
|
)
|
||||||
|
# crc = downconns[1].generator.crc_inserter.crc
|
||||||
|
# crc = dut.double_buffer.crc
|
||||||
|
# print(
|
||||||
|
# f"CYCLE#{i} : crc error = {yield crc.error:#X} crc value = {yield crc.value:#X}"
|
||||||
|
# f" crc data = {yield crc.data:#X} engine next = {yield crc.engine.next:#X} ce = {yield crc.ce}"
|
||||||
|
# )
|
||||||
|
|
||||||
|
|
||||||
|
# extra clk cycles
|
||||||
|
cyc = i + 1
|
||||||
|
for i in range(cyc, cyc + 30):
|
||||||
|
for x in range(len(downconns)):
|
||||||
|
# yield won't reset every cycle
|
||||||
|
yield downconns[x].sink.data.eq(0)
|
||||||
|
yield downconns[x].sink.k.eq(0)
|
||||||
|
yield downconns[x].sink.stb.eq(0)
|
||||||
|
yield downconns[x].sink.eop.eq(0)
|
||||||
|
yield
|
||||||
|
print(
|
||||||
|
f"\nCYCLE#{i} : source char = {yield source.data:#X} k = {yield source.k:#X} stb = {yield source.stb} ack = {yield source.ack} eop = {yield source.eop}"
|
||||||
|
# f" source dchar = {yield source.dchar:#X} dchar_k = {yield source.dchar_k:#X}"
|
||||||
|
f"\nCYCLE#{i} : read mask = {yield dut.crossbar .mux.sel}"
|
||||||
|
# f"\nCYCLE#{i} : stream id = {yield decoder.stream_id:#X} pak_tag = {yield decoder.pak_tag:#X}"
|
||||||
|
# f" stream_pak_size = {yield decoder.stream_pak_size:#X}"
|
||||||
|
)
|
||||||
|
assert True
|
||||||
|
|
||||||
|
|
||||||
|
def testbench():
|
||||||
|
# stream_id = 0x01
|
||||||
|
streams = [
|
||||||
|
[
|
||||||
|
{"data": 0x11111111, "k": Replicate(0, 4)},
|
||||||
|
{"data": 0xB105F00D, "k": Replicate(0, 4)},
|
||||||
|
],
|
||||||
|
[
|
||||||
|
{"data": 0x22222222, "k": Replicate(0, 4)},
|
||||||
|
{"data": 0xC001BEA0, "k": Replicate(0, 4)},
|
||||||
|
],
|
||||||
|
[
|
||||||
|
{"data": 0x33333333, "k": Replicate(0, 4)},
|
||||||
|
{"data": 0xC0A79AE5, "k": Replicate(0, 4)},
|
||||||
|
],
|
||||||
|
]
|
||||||
|
|
||||||
|
packet = []
|
||||||
|
for i, s in enumerate(streams):
|
||||||
|
s[-1]["eop"] = 0
|
||||||
|
packet += [
|
||||||
|
{"data": Replicate(C(i % 2, char_width), 4), "k": Replicate(0, 4)},
|
||||||
|
{"data": Replicate(C(i, char_width), 4), "k": Replicate(0, 4)},
|
||||||
|
{
|
||||||
|
"data": Replicate(C(len(s) >> 8 & 0xFF, char_width), 4),
|
||||||
|
"k": Replicate(0, 4),
|
||||||
|
},
|
||||||
|
{"data": Replicate(C(len(s) & 0xFF, char_width), 4), "k": Replicate(0, 4)},
|
||||||
|
*s,
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
yield from check_case(packet)
|
||||||
|
|
||||||
|
|
||||||
|
run_simulation(dut, testbench(), vcd_name="sim-cxp.vcd")
|
Loading…
Reference in New Issue
Block a user