forked from M-Labs/artiq-zynq
morgan
f6b13f271d
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
324 lines
16 KiB
Python
324 lines
16 KiB
Python
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")
|