* Initial plan * Fix max_decode_depth to properly control decoder hierarchy and port generation Co-authored-by: arnavsacheti <36746504+arnavsacheti@users.noreply.github.com> * Fix test that relied on old depth behavior Co-authored-by: arnavsacheti <36746504+arnavsacheti@users.noreply.github.com> * Update documentation for max_decode_depth parameter Co-authored-by: arnavsacheti <36746504+arnavsacheti@users.noreply.github.com> * fix format * Add variable_depth RDL file and smoke tests for max_decode_depth parameter Co-authored-by: arnavsacheti <36746504+arnavsacheti@users.noreply.github.com> * Add variable depth tests for APB3 and AXI4-Lite CPUIFs Co-authored-by: arnavsacheti <36746504+arnavsacheti@users.noreply.github.com> * fix * fix * bump --------- Co-authored-by: copilot-swe-agent[bot] <198982749+Copilot@users.noreply.github.com> Co-authored-by: arnavsacheti <36746504+arnavsacheti@users.noreply.github.com>
272 lines
8.7 KiB
Python
272 lines
8.7 KiB
Python
"""AXI4-Lite smoke tests for variable depth design testing max_decode_depth parameter."""
|
|
|
|
import cocotb
|
|
from cocotb.triggers import Timer
|
|
|
|
|
|
class _AxilSlaveShim:
|
|
def __init__(self, dut):
|
|
prefix = "s_axil"
|
|
self.AWREADY = getattr(dut, f"{prefix}_AWREADY")
|
|
self.AWVALID = getattr(dut, f"{prefix}_AWVALID")
|
|
self.AWADDR = getattr(dut, f"{prefix}_AWADDR")
|
|
self.AWPROT = getattr(dut, f"{prefix}_AWPROT")
|
|
self.WREADY = getattr(dut, f"{prefix}_WREADY")
|
|
self.WVALID = getattr(dut, f"{prefix}_WVALID")
|
|
self.WDATA = getattr(dut, f"{prefix}_WDATA")
|
|
self.WSTRB = getattr(dut, f"{prefix}_WSTRB")
|
|
self.BREADY = getattr(dut, f"{prefix}_BREADY")
|
|
self.BVALID = getattr(dut, f"{prefix}_BVALID")
|
|
self.BRESP = getattr(dut, f"{prefix}_BRESP")
|
|
self.ARREADY = getattr(dut, f"{prefix}_ARREADY")
|
|
self.ARVALID = getattr(dut, f"{prefix}_ARVALID")
|
|
self.ARADDR = getattr(dut, f"{prefix}_ARADDR")
|
|
self.ARPROT = getattr(dut, f"{prefix}_ARPROT")
|
|
self.RREADY = getattr(dut, f"{prefix}_RREADY")
|
|
self.RVALID = getattr(dut, f"{prefix}_RVALID")
|
|
self.RDATA = getattr(dut, f"{prefix}_RDATA")
|
|
self.RRESP = getattr(dut, f"{prefix}_RRESP")
|
|
|
|
|
|
class _AxilMasterShim:
|
|
def __init__(self, dut, base: str):
|
|
self.AWREADY = getattr(dut, f"{base}_AWREADY")
|
|
self.AWVALID = getattr(dut, f"{base}_AWVALID")
|
|
self.AWADDR = getattr(dut, f"{base}_AWADDR")
|
|
self.AWPROT = getattr(dut, f"{base}_AWPROT")
|
|
self.WREADY = getattr(dut, f"{base}_WREADY")
|
|
self.WVALID = getattr(dut, f"{base}_WVALID")
|
|
self.WDATA = getattr(dut, f"{base}_WDATA")
|
|
self.WSTRB = getattr(dut, f"{base}_WSTRB")
|
|
self.BREADY = getattr(dut, f"{base}_BREADY")
|
|
self.BVALID = getattr(dut, f"{base}_BVALID")
|
|
self.BRESP = getattr(dut, f"{base}_BRESP")
|
|
self.ARREADY = getattr(dut, f"{base}_ARREADY")
|
|
self.ARVALID = getattr(dut, f"{base}_ARVALID")
|
|
self.ARADDR = getattr(dut, f"{base}_ARADDR")
|
|
self.ARPROT = getattr(dut, f"{base}_ARPROT")
|
|
self.RREADY = getattr(dut, f"{base}_RREADY")
|
|
self.RVALID = getattr(dut, f"{base}_RVALID")
|
|
self.RDATA = getattr(dut, f"{base}_RDATA")
|
|
self.RRESP = getattr(dut, f"{base}_RRESP")
|
|
|
|
|
|
def _axil_slave(dut):
|
|
return getattr(dut, "s_axil", None) or _AxilSlaveShim(dut)
|
|
|
|
|
|
def _axil_master(dut, base: str):
|
|
return getattr(dut, base, None) or _AxilMasterShim(dut, base)
|
|
|
|
|
|
@cocotb.test()
|
|
async def test_depth_1(dut):
|
|
"""Test max_decode_depth=1 - should have interface for inner1 only."""
|
|
s_axil = _axil_slave(dut)
|
|
|
|
# At depth 1, we should have m_axil_inner1 but not deeper interfaces
|
|
inner1 = _axil_master(dut, "m_axil_inner1")
|
|
|
|
# Default slave side inputs
|
|
s_axil.AWVALID.value = 0
|
|
s_axil.AWADDR.value = 0
|
|
s_axil.AWPROT.value = 0
|
|
s_axil.WVALID.value = 0
|
|
s_axil.WDATA.value = 0
|
|
s_axil.WSTRB.value = 0
|
|
s_axil.BREADY.value = 0
|
|
s_axil.ARVALID.value = 0
|
|
s_axil.ARADDR.value = 0
|
|
s_axil.ARPROT.value = 0
|
|
s_axil.RREADY.value = 0
|
|
|
|
inner1.AWREADY.value = 0
|
|
inner1.WREADY.value = 0
|
|
inner1.BVALID.value = 0
|
|
inner1.BRESP.value = 0
|
|
inner1.ARREADY.value = 0
|
|
inner1.RVALID.value = 0
|
|
inner1.RDATA.value = 0
|
|
inner1.RRESP.value = 0
|
|
|
|
await Timer(1, units="ns")
|
|
|
|
# Write to address 0x0 (should select inner1)
|
|
inner1.AWREADY.value = 1
|
|
inner1.WREADY.value = 1
|
|
s_axil.AWVALID.value = 1
|
|
s_axil.AWADDR.value = 0x0
|
|
s_axil.WVALID.value = 1
|
|
s_axil.WDATA.value = 0x12345678
|
|
s_axil.WSTRB.value = 0xF
|
|
|
|
await Timer(1, units="ns")
|
|
|
|
assert int(inner1.AWVALID.value) == 1, "inner1 write address valid must be set"
|
|
assert int(inner1.WVALID.value) == 1, "inner1 write data valid must be set"
|
|
|
|
|
|
@cocotb.test()
|
|
async def test_depth_2(dut):
|
|
"""Test max_decode_depth=2 - should have interfaces for reg1 and inner2."""
|
|
s_axil = _axil_slave(dut)
|
|
|
|
# At depth 2, we should have m_axil_reg1 and m_axil_inner2
|
|
reg1 = _axil_master(dut, "m_axil_reg1")
|
|
inner2 = _axil_master(dut, "m_axil_inner2")
|
|
|
|
# Default slave side inputs
|
|
s_axil.AWVALID.value = 0
|
|
s_axil.AWADDR.value = 0
|
|
s_axil.AWPROT.value = 0
|
|
s_axil.WVALID.value = 0
|
|
s_axil.WDATA.value = 0
|
|
s_axil.WSTRB.value = 0
|
|
s_axil.BREADY.value = 0
|
|
s_axil.ARVALID.value = 0
|
|
s_axil.ARADDR.value = 0
|
|
s_axil.ARPROT.value = 0
|
|
s_axil.RREADY.value = 0
|
|
|
|
for master in [reg1, inner2]:
|
|
master.AWREADY.value = 0
|
|
master.WREADY.value = 0
|
|
master.BVALID.value = 0
|
|
master.BRESP.value = 0
|
|
master.ARREADY.value = 0
|
|
master.RVALID.value = 0
|
|
master.RDATA.value = 0
|
|
master.RRESP.value = 0
|
|
|
|
await Timer(1, units="ns")
|
|
|
|
# Write to address 0x0 (should select reg1)
|
|
reg1.AWREADY.value = 1
|
|
reg1.WREADY.value = 1
|
|
s_axil.AWVALID.value = 1
|
|
s_axil.AWADDR.value = 0x0
|
|
s_axil.WVALID.value = 1
|
|
s_axil.WDATA.value = 0xABCDEF01
|
|
s_axil.WSTRB.value = 0xF
|
|
|
|
await Timer(1, units="ns")
|
|
|
|
assert int(reg1.AWVALID.value) == 1, "reg1 must be selected for address 0x0"
|
|
assert int(inner2.AWVALID.value) == 0, "inner2 should not be selected"
|
|
|
|
# Reset
|
|
s_axil.AWVALID.value = 0
|
|
s_axil.WVALID.value = 0
|
|
reg1.AWREADY.value = 0
|
|
reg1.WREADY.value = 0
|
|
await Timer(1, units="ns")
|
|
|
|
# Write to address 0x10 (should select inner2)
|
|
inner2.AWREADY.value = 1
|
|
inner2.WREADY.value = 1
|
|
s_axil.AWVALID.value = 1
|
|
s_axil.AWADDR.value = 0x10
|
|
s_axil.WVALID.value = 1
|
|
s_axil.WDATA.value = 0x23456789
|
|
s_axil.WSTRB.value = 0xF
|
|
|
|
await Timer(1, units="ns")
|
|
|
|
assert int(inner2.AWVALID.value) == 1, "inner2 must be selected for address 0x10"
|
|
assert int(reg1.AWVALID.value) == 0, "reg1 should not be selected"
|
|
|
|
|
|
@cocotb.test()
|
|
async def test_depth_0(dut):
|
|
"""Test max_decode_depth=0 - should have interfaces for all leaf registers."""
|
|
s_axil = _axil_slave(dut)
|
|
|
|
# At depth 0, we should have all leaf registers: reg1, reg2, reg2b
|
|
reg1 = _axil_master(dut, "m_axil_reg1")
|
|
reg2 = _axil_master(dut, "m_axil_reg2")
|
|
reg2b = _axil_master(dut, "m_axil_reg2b")
|
|
|
|
# Default slave side inputs
|
|
s_axil.AWVALID.value = 0
|
|
s_axil.AWADDR.value = 0
|
|
s_axil.AWPROT.value = 0
|
|
s_axil.WVALID.value = 0
|
|
s_axil.WDATA.value = 0
|
|
s_axil.WSTRB.value = 0
|
|
s_axil.BREADY.value = 0
|
|
s_axil.ARVALID.value = 0
|
|
s_axil.ARADDR.value = 0
|
|
s_axil.ARPROT.value = 0
|
|
s_axil.RREADY.value = 0
|
|
|
|
for master in [reg1, reg2, reg2b]:
|
|
master.AWREADY.value = 0
|
|
master.WREADY.value = 0
|
|
master.BVALID.value = 0
|
|
master.BRESP.value = 0
|
|
master.ARREADY.value = 0
|
|
master.RVALID.value = 0
|
|
master.RDATA.value = 0
|
|
master.RRESP.value = 0
|
|
|
|
await Timer(1, units="ns")
|
|
|
|
# Write to address 0x0 (should select reg1)
|
|
reg1.AWREADY.value = 1
|
|
reg1.WREADY.value = 1
|
|
s_axil.AWVALID.value = 1
|
|
s_axil.AWADDR.value = 0x0
|
|
s_axil.WVALID.value = 1
|
|
s_axil.WDATA.value = 0x11111111
|
|
s_axil.WSTRB.value = 0xF
|
|
|
|
await Timer(1, units="ns")
|
|
|
|
assert int(reg1.AWVALID.value) == 1, "reg1 must be selected for address 0x0"
|
|
assert int(reg2.AWVALID.value) == 0, "reg2 should not be selected"
|
|
assert int(reg2b.AWVALID.value) == 0, "reg2b should not be selected"
|
|
|
|
# Reset
|
|
s_axil.AWVALID.value = 0
|
|
s_axil.WVALID.value = 0
|
|
reg1.AWREADY.value = 0
|
|
reg1.WREADY.value = 0
|
|
await Timer(1, units="ns")
|
|
|
|
# Write to address 0x10 (should select reg2)
|
|
reg2.AWREADY.value = 1
|
|
reg2.WREADY.value = 1
|
|
s_axil.AWVALID.value = 1
|
|
s_axil.AWADDR.value = 0x10
|
|
s_axil.WVALID.value = 1
|
|
s_axil.WDATA.value = 0x22222222
|
|
s_axil.WSTRB.value = 0xF
|
|
|
|
await Timer(1, units="ns")
|
|
|
|
assert int(reg2.AWVALID.value) == 1, "reg2 must be selected for address 0x10"
|
|
assert int(reg1.AWVALID.value) == 0, "reg1 should not be selected"
|
|
assert int(reg2b.AWVALID.value) == 0, "reg2b should not be selected"
|
|
|
|
# Reset
|
|
s_axil.AWVALID.value = 0
|
|
s_axil.WVALID.value = 0
|
|
reg2.AWREADY.value = 0
|
|
reg2.WREADY.value = 0
|
|
await Timer(1, units="ns")
|
|
|
|
# Write to address 0x14 (should select reg2b)
|
|
reg2b.AWREADY.value = 1
|
|
reg2b.WREADY.value = 1
|
|
s_axil.AWVALID.value = 1
|
|
s_axil.AWADDR.value = 0x14
|
|
s_axil.WVALID.value = 1
|
|
s_axil.WDATA.value = 0x33333333
|
|
s_axil.WSTRB.value = 0xF
|
|
|
|
await Timer(1, units="ns")
|
|
|
|
assert int(reg2b.AWVALID.value) == 1, "reg2b must be selected for address 0x14"
|
|
assert int(reg1.AWVALID.value) == 0, "reg1 should not be selected"
|
|
assert int(reg2.AWVALID.value) == 0, "reg2 should not be selected"
|