|
|
|
|
@@ -2,14 +2,20 @@
|
|
|
|
|
|
|
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
|
|
import json
|
|
|
|
|
import os
|
|
|
|
|
from typing import Any, Iterable
|
|
|
|
|
from typing import Any
|
|
|
|
|
|
|
|
|
|
import cocotb
|
|
|
|
|
from cocotb.triggers import Timer
|
|
|
|
|
from cocotb.triggers import RisingEdge, Timer
|
|
|
|
|
|
|
|
|
|
from tests.cocotb_lib.handle_utils import SignalHandle, resolve_handle
|
|
|
|
|
from tests.cocotb_lib.handle_utils import SignalHandle
|
|
|
|
|
from tests.cocotb_lib.protocol_utils import (
|
|
|
|
|
all_index_pairs,
|
|
|
|
|
find_invalid_address,
|
|
|
|
|
get_int,
|
|
|
|
|
load_config,
|
|
|
|
|
set_value,
|
|
|
|
|
start_clock,
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class _Apb3SlaveShim:
|
|
|
|
|
@@ -17,6 +23,8 @@ class _Apb3SlaveShim:
|
|
|
|
|
|
|
|
|
|
def __init__(self, dut):
|
|
|
|
|
prefix = "s_apb"
|
|
|
|
|
self.PCLK = getattr(dut, f"{prefix}_PCLK", None)
|
|
|
|
|
self.PRESETn = getattr(dut, f"{prefix}_PRESETn", None)
|
|
|
|
|
self.PSEL = getattr(dut, f"{prefix}_PSEL")
|
|
|
|
|
self.PENABLE = getattr(dut, f"{prefix}_PENABLE")
|
|
|
|
|
self.PWRITE = getattr(dut, f"{prefix}_PWRITE")
|
|
|
|
|
@@ -27,25 +35,6 @@ class _Apb3SlaveShim:
|
|
|
|
|
self.PSLVERR = getattr(dut, f"{prefix}_PSLVERR")
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _load_config() -> dict[str, Any]:
|
|
|
|
|
payload = os.environ.get("RDL_TEST_CONFIG")
|
|
|
|
|
if payload is None:
|
|
|
|
|
raise RuntimeError("RDL_TEST_CONFIG environment variable was not provided")
|
|
|
|
|
return json.loads(payload)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _resolve(handle, indices: Iterable[int]):
|
|
|
|
|
return resolve_handle(handle, indices)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _set_value(handle, indices: Iterable[int], value: int) -> None:
|
|
|
|
|
_resolve(handle, indices).value = value
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _get_int(handle, indices: Iterable[int]) -> int:
|
|
|
|
|
return int(_resolve(handle, indices).value)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _build_master_table(dut, masters_cfg: list[dict[str, Any]]) -> dict[str, dict[str, Any]]:
|
|
|
|
|
table: dict[str, dict[str, Any]] = {}
|
|
|
|
|
for master in masters_cfg:
|
|
|
|
|
@@ -71,12 +60,6 @@ def _build_master_table(dut, masters_cfg: list[dict[str, Any]]) -> dict[str, dic
|
|
|
|
|
return table
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _all_index_pairs(table: dict[str, dict[str, Any]]):
|
|
|
|
|
for name, entry in table.items():
|
|
|
|
|
for idx in entry["indices"]:
|
|
|
|
|
yield name, idx
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _write_pattern(address: int, width: int) -> int:
|
|
|
|
|
mask = (1 << width) - 1
|
|
|
|
|
return ((address * 0x2041) ^ 0xCAFEBABE) & mask
|
|
|
|
|
@@ -90,23 +73,30 @@ def _read_pattern(address: int, width: int) -> int:
|
|
|
|
|
@cocotb.test()
|
|
|
|
|
async def test_apb3_address_decoding(dut) -> None:
|
|
|
|
|
"""Exercise the APB3 slave interface against sampled register addresses."""
|
|
|
|
|
config = _load_config()
|
|
|
|
|
config = load_config()
|
|
|
|
|
slave = _Apb3SlaveShim(dut)
|
|
|
|
|
masters = _build_master_table(dut, config["masters"])
|
|
|
|
|
|
|
|
|
|
await start_clock(slave.PCLK)
|
|
|
|
|
if slave.PRESETn is not None:
|
|
|
|
|
slave.PRESETn.value = 1
|
|
|
|
|
|
|
|
|
|
slave.PSEL.value = 0
|
|
|
|
|
slave.PENABLE.value = 0
|
|
|
|
|
slave.PWRITE.value = 0
|
|
|
|
|
slave.PADDR.value = 0
|
|
|
|
|
slave.PWDATA.value = 0
|
|
|
|
|
|
|
|
|
|
for master_name, idx in _all_index_pairs(masters):
|
|
|
|
|
for master_name, idx in all_index_pairs(masters):
|
|
|
|
|
entry = masters[master_name]
|
|
|
|
|
_set_value(entry["inputs"]["PRDATA"], idx, 0)
|
|
|
|
|
_set_value(entry["inputs"]["PREADY"], idx, 0)
|
|
|
|
|
_set_value(entry["inputs"]["PSLVERR"], idx, 0)
|
|
|
|
|
set_value(entry["inputs"]["PRDATA"], idx, 0)
|
|
|
|
|
set_value(entry["inputs"]["PREADY"], idx, 0)
|
|
|
|
|
set_value(entry["inputs"]["PSLVERR"], idx, 0)
|
|
|
|
|
|
|
|
|
|
await Timer(1, unit="ns")
|
|
|
|
|
if slave.PCLK is not None:
|
|
|
|
|
await RisingEdge(slave.PCLK)
|
|
|
|
|
else:
|
|
|
|
|
await Timer(1, unit="ns")
|
|
|
|
|
|
|
|
|
|
addr_mask = (1 << config["address_width"]) - 1
|
|
|
|
|
|
|
|
|
|
@@ -118,85 +108,203 @@ async def test_apb3_address_decoding(dut) -> None:
|
|
|
|
|
address = txn["address"] & addr_mask
|
|
|
|
|
write_data = _write_pattern(address, config["data_width"])
|
|
|
|
|
|
|
|
|
|
_set_value(entry["inputs"]["PREADY"], index, 1)
|
|
|
|
|
_set_value(entry["inputs"]["PSLVERR"], index, 0)
|
|
|
|
|
set_value(entry["inputs"]["PREADY"], index, 0)
|
|
|
|
|
set_value(entry["inputs"]["PSLVERR"], index, 0)
|
|
|
|
|
|
|
|
|
|
# ------------------------------------------------------------------
|
|
|
|
|
# Setup phase
|
|
|
|
|
# ------------------------------------------------------------------
|
|
|
|
|
slave.PADDR.value = address
|
|
|
|
|
slave.PWDATA.value = write_data
|
|
|
|
|
slave.PWRITE.value = 1
|
|
|
|
|
slave.PSEL.value = 1
|
|
|
|
|
slave.PENABLE.value = 1
|
|
|
|
|
slave.PENABLE.value = 0
|
|
|
|
|
|
|
|
|
|
dut._log.info(
|
|
|
|
|
f"Starting transaction {txn['label']} to {master_name}{index} at address 0x{address:08X}"
|
|
|
|
|
)
|
|
|
|
|
master_address = (address - entry["inst_address"]) % entry["inst_size"]
|
|
|
|
|
|
|
|
|
|
await Timer(1, unit="ns")
|
|
|
|
|
if slave.PCLK is not None:
|
|
|
|
|
await RisingEdge(slave.PCLK)
|
|
|
|
|
else:
|
|
|
|
|
await Timer(1, unit="ns")
|
|
|
|
|
|
|
|
|
|
assert _get_int(entry["outputs"]["PSEL"], index) == 1, f"{master_name} should assert PSEL for write"
|
|
|
|
|
assert _get_int(entry["outputs"]["PWRITE"], index) == 1, f"{master_name} should see write direction"
|
|
|
|
|
assert _get_int(entry["outputs"]["PADDR"], index) == master_address, (
|
|
|
|
|
assert get_int(entry["outputs"]["PSEL"], index) == 1, f"{master_name} should assert PSEL for write"
|
|
|
|
|
assert get_int(entry["outputs"]["PENABLE"], index) == 0, (
|
|
|
|
|
f"{master_name} must hold PENABLE low in setup"
|
|
|
|
|
)
|
|
|
|
|
assert get_int(entry["outputs"]["PWRITE"], index) == 1, f"{master_name} should see write direction"
|
|
|
|
|
assert get_int(entry["outputs"]["PADDR"], index) == master_address, (
|
|
|
|
|
f"{master_name} must receive write address"
|
|
|
|
|
)
|
|
|
|
|
assert _get_int(entry["outputs"]["PWDATA"], index) == write_data, (
|
|
|
|
|
assert get_int(entry["outputs"]["PWDATA"], index) == write_data, (
|
|
|
|
|
f"{master_name} must receive write data"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
for other_name, other_idx in _all_index_pairs(masters):
|
|
|
|
|
for other_name, other_idx in all_index_pairs(masters):
|
|
|
|
|
if other_name == master_name and other_idx == index:
|
|
|
|
|
continue
|
|
|
|
|
other_entry = masters[other_name]
|
|
|
|
|
assert _get_int(other_entry["outputs"]["PSEL"], other_idx) == 0, (
|
|
|
|
|
assert get_int(other_entry["outputs"]["PSEL"], other_idx) == 0, (
|
|
|
|
|
f"{other_name}{other_idx} should remain idle during {txn['label']}"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# ------------------------------------------------------------------
|
|
|
|
|
# Access phase
|
|
|
|
|
# ------------------------------------------------------------------
|
|
|
|
|
set_value(entry["inputs"]["PREADY"], index, 1)
|
|
|
|
|
slave.PENABLE.value = 1
|
|
|
|
|
|
|
|
|
|
if slave.PCLK is not None:
|
|
|
|
|
await RisingEdge(slave.PCLK)
|
|
|
|
|
else:
|
|
|
|
|
await Timer(1, unit="ns")
|
|
|
|
|
|
|
|
|
|
assert get_int(entry["outputs"]["PSEL"], index) == 1, f"{master_name} must keep PSEL asserted"
|
|
|
|
|
assert get_int(entry["outputs"]["PENABLE"], index) == 1, (
|
|
|
|
|
f"{master_name} must assert PENABLE in access"
|
|
|
|
|
)
|
|
|
|
|
assert get_int(entry["outputs"]["PADDR"], index) == master_address, (
|
|
|
|
|
f"{master_name} must keep write address stable"
|
|
|
|
|
)
|
|
|
|
|
assert get_int(entry["outputs"]["PWDATA"], index) == write_data, (
|
|
|
|
|
f"{master_name} must keep write data stable"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
assert int(slave.PREADY.value) == 1, "Slave ready should mirror selected master"
|
|
|
|
|
assert int(slave.PSLVERR.value) == 0, "Write should complete without error"
|
|
|
|
|
|
|
|
|
|
slave.PSEL.value = 0
|
|
|
|
|
slave.PENABLE.value = 0
|
|
|
|
|
slave.PWRITE.value = 0
|
|
|
|
|
_set_value(entry["inputs"]["PREADY"], index, 0)
|
|
|
|
|
await Timer(1, unit="ns")
|
|
|
|
|
set_value(entry["inputs"]["PREADY"], index, 0)
|
|
|
|
|
if slave.PCLK is not None:
|
|
|
|
|
await RisingEdge(slave.PCLK)
|
|
|
|
|
else:
|
|
|
|
|
await Timer(1, unit="ns")
|
|
|
|
|
|
|
|
|
|
# ------------------------------------------------------------------
|
|
|
|
|
# Read phase
|
|
|
|
|
# ------------------------------------------------------------------
|
|
|
|
|
read_data = _read_pattern(address, config["data_width"])
|
|
|
|
|
_set_value(entry["inputs"]["PRDATA"], index, read_data)
|
|
|
|
|
_set_value(entry["inputs"]["PREADY"], index, 1)
|
|
|
|
|
_set_value(entry["inputs"]["PSLVERR"], index, 0)
|
|
|
|
|
set_value(entry["inputs"]["PRDATA"], index, read_data)
|
|
|
|
|
set_value(entry["inputs"]["PREADY"], index, 0)
|
|
|
|
|
set_value(entry["inputs"]["PSLVERR"], index, 0)
|
|
|
|
|
|
|
|
|
|
# ------------------------------------------------------------------
|
|
|
|
|
# Setup phase
|
|
|
|
|
# ------------------------------------------------------------------
|
|
|
|
|
slave.PADDR.value = address
|
|
|
|
|
slave.PWRITE.value = 0
|
|
|
|
|
slave.PSEL.value = 1
|
|
|
|
|
slave.PENABLE.value = 1
|
|
|
|
|
slave.PENABLE.value = 0
|
|
|
|
|
|
|
|
|
|
await Timer(1, unit="ns")
|
|
|
|
|
if slave.PCLK is not None:
|
|
|
|
|
await RisingEdge(slave.PCLK)
|
|
|
|
|
else:
|
|
|
|
|
await Timer(1, unit="ns")
|
|
|
|
|
|
|
|
|
|
assert _get_int(entry["outputs"]["PSEL"], index) == 1, f"{master_name} must assert PSEL for read"
|
|
|
|
|
assert _get_int(entry["outputs"]["PWRITE"], index) == 0, (
|
|
|
|
|
assert get_int(entry["outputs"]["PSEL"], index) == 1, f"{master_name} must assert PSEL for read"
|
|
|
|
|
assert get_int(entry["outputs"]["PENABLE"], index) == 0, (
|
|
|
|
|
f"{master_name} must hold PENABLE low in setup"
|
|
|
|
|
)
|
|
|
|
|
assert get_int(entry["outputs"]["PWRITE"], index) == 0, (
|
|
|
|
|
f"{master_name} should clear write during read"
|
|
|
|
|
)
|
|
|
|
|
assert _get_int(entry["outputs"]["PADDR"], index) == master_address, (
|
|
|
|
|
assert get_int(entry["outputs"]["PADDR"], index) == master_address, (
|
|
|
|
|
f"{master_name} must receive read address"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
for other_name, other_idx in _all_index_pairs(masters):
|
|
|
|
|
for other_name, other_idx in all_index_pairs(masters):
|
|
|
|
|
if other_name == master_name and other_idx == index:
|
|
|
|
|
continue
|
|
|
|
|
other_entry = masters[other_name]
|
|
|
|
|
assert _get_int(other_entry["outputs"]["PSEL"], other_idx) == 0, (
|
|
|
|
|
assert get_int(other_entry["outputs"]["PSEL"], other_idx) == 0, (
|
|
|
|
|
f"{other_name}{other_idx} must stay idle during read of {txn['label']}"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# ------------------------------------------------------------------
|
|
|
|
|
# Access phase
|
|
|
|
|
# ------------------------------------------------------------------
|
|
|
|
|
set_value(entry["inputs"]["PREADY"], index, 1)
|
|
|
|
|
slave.PENABLE.value = 1
|
|
|
|
|
|
|
|
|
|
if slave.PCLK is not None:
|
|
|
|
|
await RisingEdge(slave.PCLK)
|
|
|
|
|
else:
|
|
|
|
|
await Timer(1, unit="ns")
|
|
|
|
|
|
|
|
|
|
assert get_int(entry["outputs"]["PSEL"], index) == 1, f"{master_name} must keep PSEL asserted"
|
|
|
|
|
assert get_int(entry["outputs"]["PENABLE"], index) == 1, (
|
|
|
|
|
f"{master_name} must assert PENABLE in access"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
assert int(slave.PRDATA.value) == read_data, "Read data should propagate back to the slave"
|
|
|
|
|
assert int(slave.PREADY.value) == 1, "Slave ready should acknowledge the read"
|
|
|
|
|
assert int(slave.PSLVERR.value) == 0, "Read should not signal an error"
|
|
|
|
|
|
|
|
|
|
slave.PSEL.value = 0
|
|
|
|
|
slave.PENABLE.value = 0
|
|
|
|
|
_set_value(entry["inputs"]["PREADY"], index, 0)
|
|
|
|
|
_set_value(entry["inputs"]["PRDATA"], index, 0)
|
|
|
|
|
set_value(entry["inputs"]["PREADY"], index, 0)
|
|
|
|
|
set_value(entry["inputs"]["PRDATA"], index, 0)
|
|
|
|
|
if slave.PCLK is not None:
|
|
|
|
|
await RisingEdge(slave.PCLK)
|
|
|
|
|
else:
|
|
|
|
|
await Timer(1, unit="ns")
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@cocotb.test()
|
|
|
|
|
async def test_apb3_invalid_address_response(dut) -> None:
|
|
|
|
|
"""Ensure invalid addresses yield an error response and no master select."""
|
|
|
|
|
config = load_config()
|
|
|
|
|
slave = _Apb3SlaveShim(dut)
|
|
|
|
|
masters = _build_master_table(dut, config["masters"])
|
|
|
|
|
|
|
|
|
|
await start_clock(slave.PCLK)
|
|
|
|
|
if slave.PRESETn is not None:
|
|
|
|
|
slave.PRESETn.value = 1
|
|
|
|
|
|
|
|
|
|
slave.PSEL.value = 0
|
|
|
|
|
slave.PENABLE.value = 0
|
|
|
|
|
slave.PWRITE.value = 0
|
|
|
|
|
slave.PADDR.value = 0
|
|
|
|
|
slave.PWDATA.value = 0
|
|
|
|
|
|
|
|
|
|
for master_name, idx in all_index_pairs(masters):
|
|
|
|
|
entry = masters[master_name]
|
|
|
|
|
set_value(entry["inputs"]["PREADY"], idx, 0)
|
|
|
|
|
set_value(entry["inputs"]["PSLVERR"], idx, 0)
|
|
|
|
|
set_value(entry["inputs"]["PRDATA"], idx, 0)
|
|
|
|
|
|
|
|
|
|
invalid_addr = find_invalid_address(config)
|
|
|
|
|
if invalid_addr is None:
|
|
|
|
|
dut._log.warning("No unmapped address found; skipping invalid address test")
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
slave.PADDR.value = invalid_addr
|
|
|
|
|
slave.PWRITE.value = 1
|
|
|
|
|
slave.PWDATA.value = _write_pattern(invalid_addr, config["data_width"])
|
|
|
|
|
slave.PSEL.value = 1
|
|
|
|
|
slave.PENABLE.value = 0
|
|
|
|
|
|
|
|
|
|
if slave.PCLK is not None:
|
|
|
|
|
await RisingEdge(slave.PCLK)
|
|
|
|
|
else:
|
|
|
|
|
await Timer(1, unit="ns")
|
|
|
|
|
|
|
|
|
|
slave.PENABLE.value = 1
|
|
|
|
|
|
|
|
|
|
if slave.PCLK is not None:
|
|
|
|
|
await RisingEdge(slave.PCLK)
|
|
|
|
|
else:
|
|
|
|
|
await Timer(1, unit="ns")
|
|
|
|
|
|
|
|
|
|
for master_name, idx in all_index_pairs(masters):
|
|
|
|
|
entry = masters[master_name]
|
|
|
|
|
assert get_int(entry["outputs"]["PSEL"], idx) == 0, (
|
|
|
|
|
f"{master_name}{idx} must stay idle for invalid address"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
assert int(slave.PREADY.value) == 1, "Invalid address should still complete the transfer"
|
|
|
|
|
assert int(slave.PSLVERR.value) == 1, "Invalid address should raise PSLVERR"
|
|
|
|
|
|