From 6d8bba8dad67f1cb9a884762ef495bcbd13850da Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jakub=20Urba=C5=84czyk?= Date: Tue, 12 Mar 2024 16:49:18 +0000 Subject: [PATCH 1/7] Add B extension support to the toolchain Dockerfile (#615) --- docker/riscv-toolchain.Dockerfile | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/docker/riscv-toolchain.Dockerfile b/docker/riscv-toolchain.Dockerfile index 957141eb0..a998e79e3 100644 --- a/docker/riscv-toolchain.Dockerfile +++ b/docker/riscv-toolchain.Dockerfile @@ -12,8 +12,8 @@ RUN apt-get update && \ RUN git clone --shallow-since=2023.05.01 https://github.com/riscv/riscv-gnu-toolchain && \ cd riscv-gnu-toolchain && \ - git checkout 2023.05.14 && \ - ./configure --with-multilib-generator="rv32i-ilp32--a*zifence*zicsr;rv32im-ilp32--a*zifence*zicsr;rv32ic-ilp32--a*zifence*zicsr;rv32imc-ilp32--a*zifence*zicsr;rv32imfc-ilp32f--a*zifence;rv32i_zmmul-ilp32--a*zifence*zicsr;rv32ic_zmmul-ilp32--a*zifence*zicsr" && \ + git checkout 2023.12.10 && \ + ./configure --with-multilib-generator="rv32i-ilp32--a*zifence*zicsr;rv32im-ilp32--a*zifence*zicsr;rv32ic-ilp32--a*zifence*zicsr;rv32imc-ilp32--a*zifence*zicsr;rv32imfc-ilp32f--a*zifence;rv32imc_zba_zbb_zbc_zbs-ilp32--a*zifence*zicsr" && \ make -j$(nproc) && \ cd / && rm -rf riscv-gnu-toolchain From 58249da625a3b981af7d4bafab97b47b6e73f811 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jakub=20Urba=C5=84czyk?= Date: Tue, 12 Mar 2024 17:21:11 +0000 Subject: [PATCH 2/7] Enable division and bit manipulation extensions in benchmarks (#616) --- .github/workflows/benchmark.yml | 2 +- .github/workflows/main.yml | 4 ++-- test/external/embench/board_config/coreblocks-sim/board.cfg | 4 ++-- 3 files changed, 5 insertions(+), 5 deletions(-) diff --git a/.github/workflows/benchmark.yml b/.github/workflows/benchmark.yml index 45429f68d..1fc7fac1d 100644 --- a/.github/workflows/benchmark.yml +++ b/.github/workflows/benchmark.yml @@ -63,7 +63,7 @@ jobs: build-perf-benchmarks: name: Build performance benchmarks runs-on: ubuntu-latest - container: ghcr.io/kuznia-rdzeni/riscv-toolchain:2023.11.19_v + container: ghcr.io/kuznia-rdzeni/riscv-toolchain:2024.03.12 steps: - name: Checkout uses: actions/checkout@v3 diff --git a/.github/workflows/main.yml b/.github/workflows/main.yml index 5ee446967..23c1aa56c 100644 --- a/.github/workflows/main.yml +++ b/.github/workflows/main.yml @@ -48,7 +48,7 @@ jobs: build-riscof-tests: name: Build regression tests (riscv-arch-test) runs-on: ubuntu-latest - container: ghcr.io/kuznia-rdzeni/riscv-toolchain:2023.11.19_v + container: ghcr.io/kuznia-rdzeni/riscv-toolchain:2024.03.12 timeout-minutes: 10 env: PYENV_ROOT: "/root/.pyenv" @@ -180,7 +180,7 @@ jobs: build-regression-tests: name: Build regression tests (riscv-tests) runs-on: ubuntu-latest - container: ghcr.io/kuznia-rdzeni/riscv-toolchain:2023.11.19_v + container: ghcr.io/kuznia-rdzeni/riscv-toolchain:2024.03.12 timeout-minutes: 10 outputs: cache_hit: ${{ steps.cache-regression.outputs.cache-hit }} diff --git a/test/external/embench/board_config/coreblocks-sim/board.cfg b/test/external/embench/board_config/coreblocks-sim/board.cfg index 96eaae307..b1a885340 100644 --- a/test/external/embench/board_config/coreblocks-sim/board.cfg +++ b/test/external/embench/board_config/coreblocks-sim/board.cfg @@ -1,5 +1,5 @@ cc = 'riscv64-unknown-elf-gcc' -cflags = (['-c', '-fdata-sections', '-march=rv32ic_zmmul_zicsr', '-mabi=ilp32']) -ldflags = (['-Wl,-gc-sections', '-march=rv32ic_zmmul_zicsr', '-mabi=ilp32', '-nostartfiles', '-T../../../common/link.ld']) +cflags = (['-c', '-fdata-sections', '-march=rv32imc_zba_zbb_zbc_zbs_zicsr', '-mabi=ilp32']) +ldflags = (['-Wl,-gc-sections', '-march=rv32imc_zba_zbb_zbc_zbs_zicsr', '-mabi=ilp32', '-nostartfiles', '-T../../../common/link.ld']) user_libs = (['-lm']) cpu_mhz = 0.01 From 182c85a06a2e2370742cdc5104895823a55b0bb4 Mon Sep 17 00:00:00 2001 From: lekcyjna123 <34948061+lekcyjna123@users.noreply.github.com> Date: Tue, 12 Mar 2024 23:19:51 +0100 Subject: [PATCH 3/7] Fix profiles on newer yosys version (#607) --- test/regression/cocotb.py | 7 +++---- transactron/utils/gen.py | 7 ++----- 2 files changed, 5 insertions(+), 9 deletions(-) diff --git a/test/regression/cocotb.py b/test/regression/cocotb.py index 87c043688..e68c6f9ca 100644 --- a/test/regression/cocotb.py +++ b/test/regression/cocotb.py @@ -157,10 +157,9 @@ def get_cocotb_handle(self, path_components: list[str]) -> ModifiableObject: # function instead of 'getattr' - this is required by cocotb. obj = obj._id(component, extended=False) except AttributeError: - if component[0] == "\\" and component[-1] == " ": - # workaround for cocotb/verilator weirdness - # for some escaped names lookup fails, but works when unescaped - obj = obj._id(component[1:-1], extended=False) + # Try with escaped name + if component[0] != "\\" and component[-1] != " ": + obj = obj._id("\\" + component + " ", extended=False) else: raise diff --git a/transactron/utils/gen.py b/transactron/utils/gen.py index 2ff40dec2..f87706750 100644 --- a/transactron/utils/gen.py +++ b/transactron/utils/gen.py @@ -149,7 +149,7 @@ def escape_verilog_identifier(identifier: str) -> str: # The standard says how to escape a identifier, but not when. So this is # a non-exhaustive list of characters that Yosys escapes (it is used # by Amaranth when generating Verilog code). - characters_to_escape = [".", "$"] + characters_to_escape = [".", "$", "-"] for char in characters_to_escape: if char in identifier: @@ -160,10 +160,7 @@ def escape_verilog_identifier(identifier: str) -> str: def get_signal_location(signal: Signal, name_map: "SignalDict") -> list[str]: raw_location = name_map[signal] - - # Amaranth escapes identifiers when generating Verilog code, but returns non-escaped identifiers - # in the name map, so we need to escape it manually. - return [escape_verilog_identifier(component) for component in raw_location] + return raw_location def collect_metric_locations(name_map: "SignalDict") -> dict[str, MetricLocation]: From e354ae81401cb4b3fd955c8f8349a37a21598aae Mon Sep 17 00:00:00 2001 From: Marek Materzok Date: Wed, 13 Mar 2024 10:57:30 +0100 Subject: [PATCH 4/7] Migration to amaranth.lib.wiring (#609) --- coreblocks/core.py | 9 +- coreblocks/peripherals/axi_lite.py | 170 +++-- coreblocks/peripherals/wishbone.py | 199 ++--- scripts/gen_verilog.py | 19 +- scripts/synthesize.py | 23 +- stubs/amaranth/lib/wiring.pyi | 1143 ++++++++++++++++++++++++++++ test/peripherals/test_axi_lite.py | 2 +- test/peripherals/test_wishbone.py | 20 +- test/regression/pysim.py | 6 +- test/test_core.py | 11 +- transactron/utils/_typing.py | 72 +- 11 files changed, 1491 insertions(+), 183 deletions(-) create mode 100644 stubs/amaranth/lib/wiring.pyi diff --git a/coreblocks/core.py b/coreblocks/core.py index 46f0f6b91..e52ad9f2c 100644 --- a/coreblocks/core.py +++ b/coreblocks/core.py @@ -1,4 +1,5 @@ from amaranth import * +from amaranth.lib.wiring import flipped, connect from transactron.utils.dependencies import DependencyManager, DependencyContext from coreblocks.stages.func_blocks_unifier import FuncBlocksUnifier @@ -27,7 +28,7 @@ from coreblocks.stages.retirement import Retirement from coreblocks.cache.icache import ICache, ICacheBypass from coreblocks.peripherals.bus_adapter import WishboneMasterAdapter -from coreblocks.peripherals.wishbone import WishboneMaster, WishboneBus +from coreblocks.peripherals.wishbone import WishboneMaster, WishboneInterface from coreblocks.cache.refiller import SimpleCommonBusCacheRefiller from coreblocks.frontend.fetch import Fetch, UnalignedFetch from transactron.lib.transformers import MethodMap, MethodProduct @@ -38,7 +39,7 @@ class Core(Elaboratable): - def __init__(self, *, gen_params: GenParams, wb_instr_bus: WishboneBus, wb_data_bus: WishboneBus): + def __init__(self, *, gen_params: GenParams, wb_instr_bus: WishboneInterface, wb_data_bus: WishboneInterface): self.gen_params = gen_params dep_manager = DependencyContext.get() @@ -117,8 +118,8 @@ def __init__(self, *, gen_params: GenParams, wb_instr_bus: WishboneBus, wb_data_ def elaborate(self, platform): m = TModule() - m.d.comb += self.wb_master_instr.wb_master.connect(self.wb_instr_bus) - m.d.comb += self.wb_master_data.wb_master.connect(self.wb_data_bus) + connect(m, flipped(self.wb_instr_bus), self.wb_master_instr.wb_master) + connect(m, flipped(self.wb_data_bus), self.wb_master_data.wb_master) m.submodules.wb_master_instr = self.wb_master_instr m.submodules.wb_master_data = self.wb_master_data diff --git a/coreblocks/peripherals/axi_lite.py b/coreblocks/peripherals/axi_lite.py index 4059e30cd..268c396ab 100644 --- a/coreblocks/peripherals/axi_lite.py +++ b/coreblocks/peripherals/axi_lite.py @@ -1,10 +1,12 @@ +from typing import Protocol, TypeAlias, runtime_checkable from amaranth import * -from amaranth.hdl.rec import DIR_FANIN, DIR_FANOUT +from amaranth.lib.wiring import Component, Signature, In, Out from transactron import Method, def_method, TModule from transactron.core import Transaction from transactron.lib.connectors import Forwarder +from transactron.utils._typing import AbstractInterface, AbstractSignature -__all__ = ["AXILiteParameters", "AXILiteMaster"] +__all__ = ["AXILiteParameters", "AXILiteSignature", "AXILiteInterface", "AXILiteMaster"] class AXILiteParameters: @@ -24,65 +26,116 @@ def __init__(self, *, data_width: int = 64, addr_width: int = 64): self.granularity = 8 -class AXILiteLayout: - """AXI-Lite bus layout generator +class AXILiteSignature(Signature): + """AXI-Lite bus signature Parameters ---------- axil_params: AXILiteParameters - Patameters used to generate AXI-Lite layout - master: Boolean - Whether the layout should be generated for master side - (if false it's generatd for the slave side) - - Attributes - ---------- - axil_layout: Record - Record of a AXI-Lite bus. + Patameters used to generate AXI-Lite signature """ - def __init__(self, axil_params: AXILiteParameters, *, master: bool = True): - write_address = [ - ("valid", 1, DIR_FANOUT if master else DIR_FANIN), - ("rdy", 1, DIR_FANIN if master else DIR_FANOUT), - ("addr", axil_params.addr_width, DIR_FANOUT if master else DIR_FANIN), - ("prot", 3, DIR_FANOUT if master else DIR_FANIN), - ] + def __init__(self, axil_params: AXILiteParameters): + write_address = Signature( + { + "valid": Out(1), + "rdy": In(1), + "addr": Out(axil_params.addr_width), + "prot": Out(3), + } + ) - write_data = [ - ("valid", 1, DIR_FANOUT if master else DIR_FANIN), - ("rdy", 1, DIR_FANIN if master else DIR_FANOUT), - ("data", axil_params.data_width, DIR_FANOUT if master else DIR_FANIN), - ("strb", axil_params.data_width // 8, DIR_FANOUT if master else DIR_FANIN), - ] + write_data = Signature( + { + "valid": Out(1), + "rdy": In(1), + "data": Out(axil_params.data_width), + "strb": Out(axil_params.data_width // 8), + } + ) - write_response = [ - ("valid", 1, DIR_FANIN if master else DIR_FANOUT), - ("rdy", 1, DIR_FANOUT if master else DIR_FANIN), - ("resp", 2, DIR_FANIN if master else DIR_FANOUT), - ] + write_response = Signature( + { + "valid": In(1), + "rdy": Out(1), + "resp": In(2), + } + ) - read_address = [ - ("valid", 1, DIR_FANOUT if master else DIR_FANIN), - ("rdy", 1, DIR_FANIN if master else DIR_FANOUT), - ("addr", axil_params.addr_width, DIR_FANOUT if master else DIR_FANIN), - ("prot", 3, DIR_FANOUT if master else DIR_FANIN), - ] + read_address = Signature( + { + "valid": Out(1), + "rdy": In(1), + "addr": Out(axil_params.addr_width), + "prot": Out(3), + } + ) - read_data = [ - ("valid", 1, DIR_FANIN if master else DIR_FANOUT), - ("rdy", 1, DIR_FANOUT if master else DIR_FANIN), - ("data", axil_params.data_width, DIR_FANIN if master else DIR_FANOUT), - ("resp", 2, DIR_FANIN if master else DIR_FANOUT), - ] + read_data = Signature( + { + "valid": In(1), + "rdy": Out(1), + "data": In(axil_params.data_width), + "resp": In(2), + } + ) - self.axil_layout = [ - ("write_address", write_address), - ("write_data", write_data), - ("write_response", write_response), - ("read_address", read_address), - ("read_data", read_data), - ] + super().__init__( + { + "write_address": Out(write_address), + "write_data": Out(write_data), + "write_response": Out(write_response), + "read_address": Out(read_address), + "read_data": Out(read_data), + } + ) + + +class AXILiteWriteAddressInterface(AbstractInterface[AbstractSignature], Protocol): + valid: Signal + rdy: Signal + addr: Signal + prot: Signal + + +class AXILiteWriteDataInterface(AbstractInterface[AbstractSignature], Protocol): + valid: Signal + rdy: Signal + data: Signal + strb: Signal + + +class AXILiteWriteResponseInterface(AbstractInterface[AbstractSignature], Protocol): + valid: Signal + rdy: Signal + resp: Signal + + +class AXILiteReadAddressInterface(AbstractInterface[AbstractSignature], Protocol): + valid: Signal + rdy: Signal + addr: Signal + prot: Signal + + +@runtime_checkable +class AXILiteReadDataInterface(AbstractInterface[AbstractSignature], Protocol): + valid: Signal + rdy: Signal + data: Signal + resp: Signal + + +class AXILiteInterface(AbstractInterface[AbstractSignature], Protocol): + write_address: AXILiteWriteAddressInterface + write_data: AXILiteWriteDataInterface + write_response: AXILiteWriteResponseInterface + read_address: AXILiteReadAddressInterface + read_data: AXILiteReadDataInterface + + +AXILiteOutChannel: TypeAlias = AXILiteWriteAddressInterface | AXILiteWriteDataInterface | AXILiteReadAddressInterface +AXILiteInChannel: TypeAlias = AXILiteWriteResponseInterface | AXILiteReadDataInterface class AXILiteMasterMethodLayouts: @@ -137,7 +190,7 @@ def __init__(self, axil_params: AXILiteParameters): ] -class AXILiteMaster(Elaboratable): +class AXILiteMaster(Component): """AXI-Lite master interface. Parameters @@ -173,10 +226,11 @@ class AXILiteMaster(Elaboratable): Returns response state as 'wr_response_layout'. """ + axil_master: AXILiteInterface + def __init__(self, axil_params: AXILiteParameters): + super().__init__({"axil_master": Out(AXILiteSignature(axil_params))}) self.axil_params = axil_params - self.axil_layout = AXILiteLayout(self.axil_params).axil_layout - self.axil_master = Record(self.axil_layout) self.method_layouts = AXILiteMasterMethodLayouts(self.axil_params) @@ -195,7 +249,7 @@ def start_request_transaction(self, m, arg, *, channel, is_address_channel): m.d.sync += channel.strb.eq(arg.strb) m.d.sync += channel.valid.eq(1) - def state_machine_request(self, m: TModule, method: Method, *, channel: Record, request_signal: Signal): + def state_machine_request(self, m: TModule, method: Method, *, channel: AXILiteOutChannel, request_signal: Signal): with m.FSM("Idle"): with m.State("Idle"): m.d.sync += channel.valid.eq(0) @@ -212,11 +266,11 @@ def state_machine_request(self, m: TModule, method: Method, *, channel: Record, with m.Else(): m.d.comb += request_signal.eq(0) - def result_handler(self, m: TModule, forwarder: Forwarder, *, data: bool, channel: Record): + def result_handler(self, m: TModule, forwarder: Forwarder, *, channel: AXILiteInChannel): with m.If(channel.rdy & channel.valid): m.d.sync += channel.rdy.eq(forwarder.read.run) with Transaction().body(m): - if data: + if isinstance(channel, AXILiteReadDataInterface): forwarder.write(m, data=channel.data, resp=channel.resp) else: forwarder.write(m, resp=channel.resp) @@ -245,7 +299,7 @@ def _(arg): self.start_request_transaction(m, arg, channel=self.axil_master.read_address, is_address_channel=True) # read_data - self.result_handler(m, rd_forwarder, data=True, channel=self.axil_master.read_data) + self.result_handler(m, rd_forwarder, channel=self.axil_master.read_data) @def_method(m, self.rd_response) def _(): @@ -276,7 +330,7 @@ def _(arg): self.start_request_transaction(m, arg, channel=self.axil_master.write_data, is_address_channel=False) # write_response - self.result_handler(m, wr_forwarder, data=False, channel=self.axil_master.write_response) + self.result_handler(m, wr_forwarder, channel=self.axil_master.write_response) @def_method(m, self.wr_response) def _(): diff --git a/coreblocks/peripherals/wishbone.py b/coreblocks/peripherals/wishbone.py index 75f71522a..e71617682 100644 --- a/coreblocks/peripherals/wishbone.py +++ b/coreblocks/peripherals/wishbone.py @@ -1,13 +1,14 @@ from amaranth import * -from amaranth.hdl.rec import DIR_FANIN, DIR_FANOUT +from amaranth.lib.wiring import PureInterface, Signature, In, Out, Component from functools import reduce -from typing import List +from typing import Protocol, cast import operator from transactron import Method, def_method, TModule from transactron.core import Transaction from transactron.lib import AdapterTrans, BasicFifo from transactron.utils import OneHotSwitchDynamic, assign, RoundRobin +from transactron.utils._typing import AbstractInterface, AbstractSignature from transactron.lib.connectors import Forwarder from transactron.utils.transactron_helpers import make_layout @@ -31,52 +32,45 @@ def __init__(self, *, data_width: int = 64, addr_width: int = 64, granularity: i self.granularity = granularity -class WishboneLayout: - """Wishbone bus Layout generator. - - Parameters - ---------- - wb_params: WishboneParameters - Parameters used to generate Wishbone layout - master: Boolean - Whether the layout should be generated for the master side - (otherwise it's generated for the slave side) - - Attributes - ---------- - wb_layout: Record - Record of a Wishbone bus. - """ - - def __init__(self, wb_params: WishboneParameters, master=True): - self.wb_layout = [ - ("dat_r", wb_params.data_width, DIR_FANIN if master else DIR_FANOUT), - ("dat_w", wb_params.data_width, DIR_FANOUT if master else DIR_FANIN), - ("rst", 1, DIR_FANOUT if master else DIR_FANIN), - ("ack", 1, DIR_FANIN if master else DIR_FANOUT), - ("adr", wb_params.addr_width, DIR_FANOUT if master else DIR_FANIN), - ("cyc", 1, DIR_FANOUT if master else DIR_FANIN), - ("stall", 1, DIR_FANIN if master else DIR_FANOUT), - ("err", 1, DIR_FANIN if master else DIR_FANOUT), - ("lock", 1, DIR_FANOUT if master else DIR_FANIN), - ("rty", 1, DIR_FANIN if master else DIR_FANOUT), - ("sel", wb_params.data_width // wb_params.granularity, DIR_FANOUT if master else DIR_FANIN), - ("stb", 1, DIR_FANOUT if master else DIR_FANIN), - ("we", 1, DIR_FANOUT if master else DIR_FANIN), - ] +class WishboneSignature(Signature): + def __init__(self, wb_params: WishboneParameters): + super().__init__( + { + "dat_r": In(wb_params.data_width), + "dat_w": Out(wb_params.data_width), + "rst": Out(1), + "ack": In(1), + "adr": Out(wb_params.addr_width), + "cyc": Out(1), + "stall": In(1), + "err": In(1), + "lock": Out(1), + "rty": In(1), + "sel": Out(wb_params.data_width // wb_params.granularity), + "stb": Out(1), + "we": Out(1), + } + ) + def create(self, *, path: tuple[str | int, ...] = (), src_loc_at: int = 0): + """Create a WishboneInterface.""" # workaround for Sphinx problem with Amaranth docstring + return cast(WishboneInterface, PureInterface(self, path=path, src_loc_at=src_loc_at + 1)) -class WishboneBus(Record): - """Wishbone bus. - Parameters - ---------- - wb_params: WishboneParameters - Parameters for bus generation. - """ - - def __init__(self, wb_params: WishboneParameters, **kwargs): - super().__init__(WishboneLayout(wb_params).wb_layout, **kwargs) +class WishboneInterface(AbstractInterface[AbstractSignature], Protocol): + dat_r: Signal + dat_w: Signal + rst: Signal + ack: Signal + adr: Signal + cyc: Signal + stall: Signal + err: Signal + lock: Signal + rty: Signal + sel: Signal + stb: Signal + we: Signal class WishboneMasterMethodLayout: @@ -107,7 +101,7 @@ def __init__(self, wb_params: WishboneParameters): self.result_layout = make_layout(("data", wb_params.data_width), ("err", 1)) -class WishboneMaster(Elaboratable): +class WishboneMaster(Component): """Wishbone bus master interface. Parameters @@ -117,7 +111,7 @@ class WishboneMaster(Elaboratable): Attributes ---------- - wb_master: Record (like WishboneLayout) + wb_master: WishboneInterface Wishbone bus output. request: Method Transactional method to start a new Wishbone request. @@ -129,10 +123,11 @@ class WishboneMaster(Elaboratable): Returns state of request (error or success) and data (in case of read request) as `result_layout`. """ + wb_master: WishboneInterface + def __init__(self, wb_params: WishboneParameters): + super().__init__({"wb_master": Out(WishboneSignature(wb_params))}) self.wb_params = wb_params - self.wb_layout = WishboneLayout(wb_params).wb_layout - self.wb_master = Record(self.wb_layout) self.method_layouts = WishboneMasterMethodLayout(wb_params) @@ -208,7 +203,7 @@ def _(arg): return m -class PipelinedWishboneMaster(Elaboratable): +class PipelinedWishboneMaster(Component): """Pipelined Wishbone bus master interface. Parameters @@ -220,7 +215,7 @@ class PipelinedWishboneMaster(Elaboratable): Attributes ---------- - wb: Record (like WishboneLayout) + wb: WishboneInterface Wishbone bus output. request: Method Transactional method to start a new Wishbone request. @@ -234,7 +229,10 @@ class PipelinedWishboneMaster(Elaboratable): True, if there are no requests waiting for response """ + wb: WishboneInterface + def __init__(self, wb_params: WishboneParameters, *, max_req: int = 8): + super().__init__({"wb": Out(WishboneSignature(wb_params))}) self.wb_params = wb_params self.max_req = max_req @@ -244,9 +242,6 @@ def __init__(self, wb_params: WishboneParameters, *, max_req: int = 8): self.requests_finished = Signal() - self.wb_layout = WishboneLayout(wb_params).wb_layout - self.wb = Record(self.wb_layout) - def generate_method_layouts(self, wb_params: WishboneParameters): # generate method layouts locally self.request_in_layout = [ @@ -307,17 +302,17 @@ def _(arg) -> None: return m -class WishboneMuxer(Elaboratable): +class WishboneMuxer(Component): """Wishbone Muxer. Connects one master to multiple slaves. Parameters ---------- - master_wb: Record (like WishboneLayout) - Record of master inteface. - slaves: List[Record] - List of connected slaves' Wishbone Records (like WishboneLayout). + wb_params: WishboneParameters + Parameters for bus generation. + num_slaves: int + Number of slave devices to multiplex. ssel_tga: Signal Signal that selects the slave to connect. Signal width is the number of slaves and each bit coresponds to a slave. This signal is a Wishbone TGA (address tag), so it needs to be valid every time Wishbone STB @@ -326,15 +321,29 @@ class WishboneMuxer(Elaboratable): different `ssel_tga` value, all pending request have to be finished (and `stall` cleared) and there have to be one cycle delay from previouse request (to deassert the STB signal). Holding new requests should be implemented in block that controlls `ssel_tga` signal, before the Wishbone Master. + + Attributes + ---------- + master_wb: WishboneInterface + Master inteface. + slaves: list of WishboneInterface + List of connected slaves' Wishbone interfaces. """ - def __init__(self, master_wb: Record, slaves: List[Record], ssel_tga: Signal): - self.master_wb = master_wb - self.slaves = slaves + master_wb: WishboneInterface + slaves: list[WishboneInterface] + + def __init__(self, wb_params: WishboneParameters, num_slaves: int, ssel_tga: Signal): + super().__init__( + { + "master_wb": Out(WishboneSignature(wb_params)), + "slaves": In(WishboneSignature(wb_params)).array(num_slaves), + } + ) self.sselTGA = ssel_tga select_bits = ssel_tga.shape().width - assert select_bits == len(slaves) + assert select_bits == num_slaves self.txn_sel = Signal(select_bits) self.txn_sel_r = Signal(select_bits) @@ -354,10 +363,9 @@ def elaborate(self, platform): for i in range(len(self.slaves)): # connect all M->S signals except stb - m.d.comb += self.master_wb.connect( - self.slaves[i], - include=["dat_w", "rst", "cyc", "lock", "adr", "we", "sel"], - ) + # workaround for the lack of selective connecting in wiring + for n in ["dat_w", "cyc", "lock", "adr", "we", "sel", "stb"]: + m.d.comb += getattr(self.slaves[i], n).eq(getattr(self.master_wb, n)) # use stb as select m.d.comb += self.slaves[i].stb.eq(self.txn_sel[i] & self.master_wb.stb) @@ -367,12 +375,14 @@ def elaborate(self, platform): m.d.comb += self.master_wb.rty.eq(reduce(operator.or_, [self.slaves[i].rty for i in range(len(self.slaves))])) for i in OneHotSwitchDynamic(m, self.txn_sel): # mux S->M data - m.d.comb += self.master_wb.connect(self.slaves[i], include=["dat_r", "stall"]) + # workaround for the lack of selective connecting in wiring + for n in ["dat_r", "stall"]: + m.d.comb += getattr(self.master_wb, n).eq(getattr(self.slaves[i], n)) return m # connects multiple masters to one slave -class WishboneArbiter(Elaboratable): +class WishboneArbiter(Component): """Wishbone Arbiter. Connects multiple masters to one slave. @@ -380,20 +390,34 @@ class WishboneArbiter(Elaboratable): Parameters ---------- - slave_wb: Record (like WishboneLayout) - Record of slave inteface. - masters: List[Record] - List of master interface Records. + wb_params: WishboneParameters + Parameters for bus generation. + num_slaves: int + Number of master devices. + + Attributes + ---------- + slave_wb: WishboneInterface + Slave inteface. + masters: list of WishboneInterface + List of master interfaces. """ - def __init__(self, slave_wb: Record, masters: List[Record]): - self.slave_wb = slave_wb - self.masters = masters + slave_wb: WishboneInterface + masters: list[WishboneInterface] + + def __init__(self, wb_params: WishboneParameters, num_masters: int): + super().__init__( + { + "slave_wb": In(WishboneSignature(wb_params)), + "masters": Out(WishboneSignature(wb_params)).array(num_masters), + } + ) self.prev_cyc = Signal() # Amaranth round robin singals self.arb_enable = Signal() - self.req_signal = Signal(len(masters)) + self.req_signal = Signal(num_masters) def elaborate(self, platform): m = TModule() @@ -417,7 +441,9 @@ def elaborate(self, platform): m.d.comb += self.masters[i].err.eq((m.submodules.rr.grant == i) & self.slave_wb.err) m.d.comb += self.masters[i].rty.eq((m.submodules.rr.grant == i) & self.slave_wb.rty) # remaining S->M signals are shared, master will only accept response if bus termination signal is present - m.d.comb += self.masters[i].connect(self.slave_wb, include=["dat_r", "stall"]) + # workaround for the lack of selective connecting in wiring + for n in ["dat_r", "stall"]: + m.d.comb += getattr(self.masters[i], n).eq(getattr(self.slave_wb, n)) # combine reset singnal m.d.comb += self.slave_wb.rst.eq(reduce(operator.or_, [self.masters[i].rst for i in range(len(self.masters))])) @@ -426,10 +452,9 @@ def elaborate(self, platform): with m.Switch(m.submodules.rr.grant): for i in range(len(self.masters)): with m.Case(i): - m.d.comb += self.masters[i].connect( - self.slave_wb, - include=["dat_w", "cyc", "lock", "adr", "we", "sel", "stb"], - ) + # workaround for the lack of selective connecting in wiring + for n in ["dat_w", "cyc", "lock", "adr", "we", "sel", "stb"]: + m.d.comb += getattr(self.slave_wb, n).eq(getattr(self.masters[i], n)) # Disable slave when round robin is not valid at start of new request # This prevents chaning grant and muxes during Wishbone cycle @@ -439,7 +464,7 @@ def elaborate(self, platform): return m -class WishboneMemorySlave(Elaboratable): +class WishboneMemorySlave(Component): """Wishbone slave with memory Wishbone slave interface with addressable memory underneath. @@ -454,11 +479,14 @@ class WishboneMemorySlave(Elaboratable): Attributes ---------- - bus: Record (like WishboneLayout) - Wishbone bus record. + bus: WishboneInterface + Wishbone bus interface. """ + bus: WishboneInterface + def __init__(self, wb_params: WishboneParameters, **kwargs): + super().__init__({"bus": In(WishboneSignature(wb_params))}) if "width" not in kwargs: kwargs["width"] = wb_params.data_width if kwargs["width"] not in (8, 16, 32, 64): @@ -470,7 +498,6 @@ def __init__(self, wb_params: WishboneParameters, **kwargs): raise RuntimeError("Granularity has to be one of: 8, 16, 32, 64") self.mem = Memory(**kwargs) - self.bus = Record(WishboneLayout(wb_params, master=False).wb_layout) def elaborate(self, platform): m = TModule() diff --git a/scripts/gen_verilog.py b/scripts/gen_verilog.py index 5696a15cf..e9c5b8707 100755 --- a/scripts/gen_verilog.py +++ b/scripts/gen_verilog.py @@ -14,10 +14,10 @@ sys.path.insert(0, parent) from coreblocks.params.genparams import GenParams -from coreblocks.peripherals.wishbone import WishboneBus +from coreblocks.peripherals.wishbone import WishboneSignature from coreblocks.core import Core from transactron import TransactionModule -from transactron.utils import flatten_signals, DependencyManager, DependencyContext +from transactron.utils import DependencyManager, DependencyContext from transactron.utils.gen import generate_verilog from coreblocks.params.configurations import * @@ -33,8 +33,8 @@ class Top(Elaboratable): def __init__(self, gen_params): self.gp: GenParams = gen_params - self.wb_instr = WishboneBus(self.gp.wb_params, name="wb_instr") - self.wb_data = WishboneBus(self.gp.wb_params, name="wb_data") + self.wb_instr = WishboneSignature(self.gp.wb_params).create() + self.wb_data = WishboneSignature(self.gp.wb_params).create() def elaborate(self, platform: Platform): m = Module() @@ -49,9 +49,14 @@ def gen_verilog(core_config: CoreConfiguration, output_path: str): with DependencyContext(DependencyManager()): gp = GenParams(core_config) top = Top(gp) - ports = list(flatten_signals(top.wb_instr)) + list(flatten_signals(top.wb_data)) - - verilog_text, gen_info = generate_verilog(top, ports) + instr_ports: list[Signal] = [getattr(top.wb_instr, name) for name in top.wb_instr.signature.members] + data_ports: list[Signal] = [getattr(top.wb_data, name) for name in top.wb_data.signature.members] + for sig in instr_ports: + sig.name = "wb_instr__" + sig.name + for sig in data_ports: + sig.name = "wb_data__" + sig.name + + verilog_text, gen_info = generate_verilog(top, instr_ports + data_ports) gen_info.encode(f"{output_path}.json") with open(output_path, "w") as f: diff --git a/scripts/synthesize.py b/scripts/synthesize.py index 73e317507..6c5c2f7eb 100755 --- a/scripts/synthesize.py +++ b/scripts/synthesize.py @@ -7,6 +7,7 @@ from amaranth.build import Platform from amaranth import * +from amaranth.lib.wiring import Flow if __name__ == "__main__": parent = os.path.dirname(os.path.dirname(os.path.abspath(__file__))) @@ -26,7 +27,7 @@ from coreblocks.fu.zbs import ZbsComponent from transactron import TransactionModule from transactron.lib import AdapterBase, AdapterTrans -from coreblocks.peripherals.wishbone import WishboneArbiter, WishboneBus +from coreblocks.peripherals.wishbone import WishboneArbiter, WishboneInterface from constants.ecp5_platforms import ( ResourceBuilder, adapter_resources, @@ -45,7 +46,7 @@ class WishboneConnector(Elaboratable): - def __init__(self, wb: WishboneBus, number: int): + def __init__(self, wb: WishboneInterface, number: int): self.wb = wb self.number = number @@ -55,7 +56,12 @@ def elaborate(self, platform: Platform): pins = platform.request("wishbone", self.number) assert isinstance(pins, Record) - m.d.comb += self.wb.connect(pins) + for name in self.wb.signature.members: + member = self.wb.signature.members[name] + if member.flow == Flow.In: + m.d.comb += getattr(pins, name).o.eq(getattr(self.wb, name)) + else: + m.d.comb += getattr(self.wb, name).eq(getattr(pins, name).i) return m @@ -93,14 +99,13 @@ def elaborate(self, platform: Platform): def unit_core(gen_params: GenParams): resources = wishbone_resources(gen_params.wb_params) - wb_instr = WishboneBus(gen_params.wb_params) - wb_data = WishboneBus(gen_params.wb_params) + wb_arbiter = WishboneArbiter(gen_params.wb_params, 2) + wb_instr = wb_arbiter.masters[0] + wb_data = wb_arbiter.masters[1] - core = Core(gen_params=gen_params, wb_instr_bus=wb_instr, wb_data_bus=wb_data) + wb_connector = WishboneConnector(wb_arbiter.slave_wb, 0) - wb = WishboneBus(gen_params.wb_params) - wb_arbiter = WishboneArbiter(wb, [wb_instr, wb_data]) - wb_connector = WishboneConnector(wb, 0) + core = Core(gen_params=gen_params, wb_instr_bus=wb_instr, wb_data_bus=wb_data) module = ModuleConnector(core=core, wb_arbiter=wb_arbiter, wb_connector=wb_connector) diff --git a/stubs/amaranth/lib/wiring.pyi b/stubs/amaranth/lib/wiring.pyi new file mode 100644 index 000000000..9565301f9 --- /dev/null +++ b/stubs/amaranth/lib/wiring.pyi @@ -0,0 +1,1143 @@ +""" +This type stub file was generated by pyright. +""" + +import enum +from collections.abc import Mapping, Iterator +from typing import NoReturn, Literal, TypeVar, Generic, Any, Self, Optional, overload +from ..hdl._ir import Elaboratable +from .._utils import final +from transactron.utils._typing import ShapeLike, ValueLike, AbstractInterface, AbstractSignature, ModuleLike + +__all__ = ["In", "Out", "Signature", "PureInterface", "connect", "flipped", "Component"] + +_T_Signature = TypeVar("_T_Signature", bound=AbstractSignature) +_T_SignatureMembers = TypeVar("_T_SignatureMembers", bound=SignatureMembers) +_T_Interface = TypeVar("_T_Interface", bound=AbstractInterface) +_T = TypeVar("_T") + +class Flow(enum.Enum): + """Direction of data flow. This enumeration has two values, :attr:`Out` and :attr:`In`, + the meaning of which depends on the context in which they are used. + """ + Out = "out" + In = "in" + def flip(self) -> Flow: + """Flip the direction of data flow. + + Returns + ------- + :class:`Flow` + :attr:`In` if called as :pc:`Out.flip()`; :attr:`Out` if called as :pc:`In.flip()`. + """ + ... + + def __call__(self, description: Signature | ShapeLike, *, reset=...) -> Member: + """Create a :class:`Member` with this data flow and the provided description and + reset value. + + Returns + ------- + :class:`Member` + :pc:`Member(self, description, reset=reset)` + """ + ... + + def __repr__(self) -> Literal['Out', 'In']: + ... + + def __str__(self) -> str: + ... + + + +Out = Flow.Out +In = Flow.In + +@final +class Member: + """Description of a signature member. + + This class is a discriminated union: its instances describe either a `port member` or + a `signature member`, and accessing properties for the wrong kind of member raises + an :exc:`AttributeError`. + + The class is created from a `description`: a :class:`Signature` instance (in which case + the :class:`Member` is created as a signature member), or a :ref:`shape-like ` + object (in which case it is created as a port member). After creation the :class:`Member` + instance cannot be modified. + + When a :class:`Signal` is created from a description of a port member, the signal's reset value + is taken from the member description. If this signal is never explicitly assigned a value, it + will equal ``reset``. + + Although instances can be created directly, most often they will be created through + :data:`In` and :data:`Out`, e.g. :pc:`In(unsigned(1))` or :pc:`Out(stream.Signature(RGBPixel))`. + """ + def __init__(self, flow: Flow, description: Signature | ShapeLike, *, reset=..., _dimensions=...) -> None: + ... + + def flip(self) -> Member: + """Flip the data flow of this member. + + Returns + ------- + :class:`Member` + A new :pc:`member` with :pc:`member.flow` equal to :pc:`self.flow.flip()`, and identical + to :pc:`self` other than that. + """ + ... + + def array(self, *dimensions) -> Member: + """Add array dimensions to this member. + + The dimensions passed to this method are `prepended` to the existing dimensions. + For example, :pc:`Out(1).array(2)` describes an array of 2 elements, whereas both + :pc:`Out(1).array(2, 3)` and :pc:`Out(1).array(3).array(2)` both describe a two dimensional + array of 2 by 3 elements. + + Dimensions are passed to :meth:`array` in the order in which they would be indexed. + That is, :pc:`.array(x, y)` creates a member that can be indexed up to :pc:`[x-1][y-1]`. + + The :meth:`array` method is composable: calling :pc:`member.array(x)` describes an array of + :pc:`x` members even if :pc:`member` was already an array. + + Returns + ------- + :class:`Member` + A new :pc:`member` with :pc:`member.dimensions` extended by :pc:`dimensions`, and + identical to :pc:`self` other than that. + """ + ... + + @property + def flow(self) -> Flow: + """Data flow of this member. + + Returns + ------- + :class:`Flow` + """ + ... + + @property + def is_port(self) -> bool: + """Whether this is a description of a port member. + + Returns + ------- + :class:`bool` + :pc:`True` if this is a description of a port member, + :pc:`False` if this is a description of a signature member. + """ + ... + + @property + def is_signature(self) -> bool: + """Whether this is a description of a signature member. + + Returns + ------- + :class:`bool` + :pc:`True` if this is a description of a signature member, + :pc:`False` if this is a description of a port member. + """ + ... + + @property + def shape(self) -> ShapeLike: + """Shape of a port member. + + Returns + ------- + :ref:`shape-like object ` + The shape that was provided when constructing this :class:`Member`. + + Raises + ------ + :exc:`AttributeError` + If :pc:`self` describes a signature member. + """ + ... + + @property + def reset(self): # -> None: + """Reset value of a port member. + + Returns + ------- + :ref:`const-castable object ` + The reset value that was provided when constructing this :class:`Member`. + + Raises + ------ + :exc:`AttributeError` + If :pc:`self` describes a signature member. + """ + ... + + @property + def signature(self) -> Signature: + """Signature of a signature member. + + Returns + ------- + :class:`Signature` + The signature that was provided when constructing this :class:`Member`. + + Raises + ------ + :exc:`AttributeError` + If :pc:`self` describes a port member. + """ + ... + + @property + def dimensions(self) -> tuple[int, ...]: + """Array dimensions. + + A member will usually have no dimensions; in this case it does not describe an array. + A single dimension describes one-dimensional array, and so on. + + Returns + ------- + :class:`tuple` of :class:`int` + Dimensions, if any, of this member, from most to least major. + """ + ... + + def __eq__(self, other) -> bool: + ... + + def __repr__(self) -> str: + ... + + + +@final +class SignatureError(Exception): + """ + This exception is raised when an invalid operation specific to signature manipulation is + performed with :class:`SignatureMembers`, such as adding a member to a frozen signature. + Other exceptions, such as :exc:`TypeError` or :exc:`NameError`, will still be raised where + appropriate. + """ + ... + + +@final +class SignatureMembers(Mapping[str, Member]): + """Mapping of signature member names to their descriptions. + + This container, a :class:`collections.abc.Mapping`, is used to implement the :pc:`members` + attribute of signature objects. + + The keys in this container must be valid Python attribute names that are public (do not begin + with an underscore. The values must be instances of :class:`Member`. The container is mutable + in a restricted manner: new keys may be added, but existing keys may not be modified or removed. + In addition, the container can be `frozen`, which disallows addition of new keys. Freezing + a container recursively freezes the members of any signatures inside. + + In addition to the use of the superscript operator, multiple members can be added at once with + the :pc:`+=` opreator. + + The :meth:`create` method converts this mapping into a mapping of names to signature members + (signals and interface objects) by creating them from their descriptions. The created mapping + can be used to populate an interface object. + """ + def __init__(self, members: Mapping[str, Member]=...) -> None: + ... + + def flip(self) -> FlippedSignatureMembers[Self]: + """Flip the data flow of the members in this mapping. + + Returns + ------- + :class:`FlippedSignatureMembers` + Proxy collection :pc:`FlippedSignatureMembers(self)` that flips the data flow of + the members that are accessed using it. + """ + ... + + def __eq__(self, other) -> bool: + """Compare the members in this and another mapping. + + Returns + ------- + :class:`bool` + :pc:`True` if the mappings contain the same key-value pairs, :pc:`False` otherwise. + """ + ... + + def __contains__(self, name: str) -> bool: + """Check whether a member with a given name exists. + + Returns + ------- + :class:`bool` + """ + ... + + def __getitem__(self, name: str) -> Member: + """Retrieves the description of a member with a given name. + + Returns + ------- + :class:`Member` + + Raises + ------ + :exc:`TypeError` + If :pc:`name` is not a string. + :exc:`NameError` + If :pc:`name` is not a valid, public Python attribute name. + :exc:`SignatureError` + If a member called :pc:`name` does not exist in the collection. + """ + ... + + def __setitem__(self, name: str, member: Member) -> NoReturn: + """Stub that forbids addition of members to the collection. + + Raises + ------ + :exc:`SignatureError` + Always. + """ + ... + + def __delitem__(self, name: str) -> NoReturn: + """Stub that forbids removal of members from the collection. + + Raises + ------ + :exc:`SignatureError` + Always. + """ + ... + + def __iter__(self) -> Iterator[str]: + """Iterate through the names of members in the collection. + + Returns + ------- + iterator of :class:`str` + Names of members, in the order of insertion. + """ + ... + + def __len__(self) -> int: + ... + + def flatten(self, *, path: tuple[str | int, ...]=...) -> Iterator[tuple[tuple[str | int, ...], Member]]: + """Recursively iterate through this collection. + + .. note:: + + The :ref:`paths ` returned by this method and by :meth:`Signature.flatten` + differ. This method yields a single result for each :class:`Member` in the collection, + disregarding their dimensions: + + .. doctest:: + + >>> sig = wiring.Signature({ + ... "items": In(1).array(2) + ... }) + >>> list(sig.members.flatten()) + [(('items',), In(1).array(2))] + + The :meth:`Signature.flatten` method yields multiple results for such a member; see + the documentation for that method for an example. + + Returns + ------- + iterator of (:class:`tuple` of :class:`str`, :class:`Member`) + Pairs of :ref:`paths ` and the corresponding members. A path yielded by + this method is a tuple of strings where each item is a key through which the item may + be reached. + """ + ... + + def create(self, *, path: tuple[str | int, ...] =..., src_loc_at: int =...) -> dict[str, Any]: + """Create members from their descriptions. + + For each port member, this function creates a :class:`Signal` with the shape and reset + value taken from the member description, and the name constructed from + the :ref:`paths ` to the member (by concatenating path items with a double + underscore, ``__``). + + For each signature member, this function calls :meth:`Signature.create` for that signature. + The resulting object can have any type if a :class:`Signature` subclass overrides + the :class:`create` method. + + If the member description includes dimensions, in each case, instead of a single member, + a :class:`list` of members is created for each dimension. (That is, for a single dimension + a list of members is returned, for two dimensions a list of lists is returned, and so on.) + + Returns + ------- + dict of :class:`str` to :ref:`value-like ` or interface object or a potentially nested list of these + Mapping of names to actual signature members. + """ + ... + + def __repr__(self) -> str: + ... + + + +@final +class FlippedSignatureMembers(Mapping[str, Member], Generic[_T_SignatureMembers]): + """Mapping of signature member names to their descriptions, with the directions flipped. + + Although an instance of :class:`FlippedSignatureMembers` could be created directly, it will + be usually created by a call to :meth:`SignatureMembers.flip`. + + This container is a wrapper around :class:`SignatureMembers` that contains the same members + as the inner mapping, but flips their data flow when they are accessed. For example: + + .. testcode:: + + members = wiring.SignatureMembers({"foo": Out(1)}) + + flipped_members = members.flip() + assert flipped_members["foo"].flow == In + + This class implements the same methods, with the same functionality (other than the flipping of + the data flow), as the :class:`SignatureMembers` class; see the documentation for that class + for details. + """ + def __init__(self, unflipped: _T_SignatureMembers) -> None: + ... + + def flip(self) -> _T_SignatureMembers: + """ + Flips this mapping back to the original one. + + Returns + ------- + :class:`SignatureMembers` + :pc:`unflipped` + """ + ... + + def __eq__(self, other) -> bool: + """Compare the members in this and another mapping. + + Returns + ------- + :class:`bool` + :pc:`True` if the mappings contain the same key-value pairs, :pc:`False` otherwise. + """ + ... + + def __contains__(self, name: str) -> bool: + ... + + def __getitem__(self, name: str) -> Member: + ... + + def __setitem__(self, name: str, member: Member) -> NoReturn: + ... + + def __delitem__(self, name: str) -> NoReturn: + ... + + def __iter__(self) -> Iterator[str]: + ... + + def __len__(self) -> int: + ... + + def flatten(self, *, path: tuple[str | int, ...] = ...) -> Iterator[tuple[tuple[str | int, ...], Member]]: + """Recursively iterate through this collection. + + .. note:: + + The :ref:`paths ` returned by this method and by :meth:`Signature.flatten` + differ. This method yields a single result for each :class:`Member` in the collection, + disregarding their dimensions: + + .. doctest:: + + >>> sig = wiring.Signature({ + ... "items": In(1).array(2) + ... }) + >>> list(sig.members.flatten()) + [(('items',), In(1).array(2))] + + The :meth:`Signature.flatten` method yields multiple results for such a member; see + the documentation for that method for an example. + + Returns + ------- + iterator of (:class:`tuple` of :class:`str`, :class:`Member`) + Pairs of :ref:`paths ` and the corresponding members. A path yielded by + this method is a tuple of strings where each item is a key through which the item may + be reached. + """ + ... + + def create(self, *, path: tuple[str | int, ...] =..., src_loc_at: int =...) -> dict[str, Any]: + """Create members from their descriptions. + + For each port member, this function creates a :class:`Signal` with the shape and reset + value taken from the member description, and the name constructed from + the :ref:`paths ` to the member (by concatenating path items with a double + underscore, ``__``). + + For each signature member, this function calls :meth:`Signature.create` for that signature. + The resulting object can have any type if a :class:`Signature` subclass overrides + the :class:`create` method. + + If the member description includes dimensions, in each case, instead of a single member, + a :class:`list` of members is created for each dimension. (That is, for a single dimension + a list of members is returned, for two dimensions a list of lists is returned, and so on.) + + Returns + ------- + dict of :class:`str` to :ref:`value-like ` or interface object or a potentially nested list of these + Mapping of names to actual signature members. + """ + ... + + def __repr__(self) -> str: + ... + + + +class SignatureMeta(type): + """Metaclass for :class:`Signature` that makes :class:`FlippedSignature` its + 'virtual subclass'. + + The object returned by :meth:`Signature.flip` is an instance of :class:`FlippedSignature`. + It implements all of the methods :class:`Signature` has, and for subclasses of + :class:`Signature`, it implements all of the methods defined on the subclass as well. + This makes it effectively a subtype of :class:`Signature` (or a derived class of it), but this + relationship is not captured by the Python type system: :class:`FlippedSignature` only has + :class:`object` as its base class. + + This metaclass extends :func:`issubclass` and :func:`isinstance` so that they take into + account the subtyping relationship between :class:`Signature` and :class:`FlippedSignature`, + described below. + """ + def __subclasscheck__(cls, subclass) -> bool: + """ + Override of :pc:`issubclass(cls, Signature)`. + + In addition to the standard behavior of :func:`issubclass`, this override makes + :class:`FlippedSignature` a subclass of :class:`Signature` or any of its subclasses. + """ + ... + + def __instancecheck__(cls, instance) -> bool: + """ + Override of :pc:`isinstance(obj, Signature)`. + + In addition to the standard behavior of :func:`isinstance`, this override makes + :pc:`isinstance(obj, cls)` act as :pc:`isinstance(obj.flip(), cls)` where + :pc:`obj` is an instance of :class:`FlippedSignature`. + """ + ... + + + +class Signature(metaclass=SignatureMeta): + """Description of an interface object. + + An interface object is a Python object that has a :pc:`signature` attribute containing + a :class:`Signature` object, as well as an attribute for every member of its signature. + Signatures and interface objects are tightly linked: an interface object can be created out + of a signature, and the signature is used when :func:`connect` ing two interface objects + together. See the :ref:`introduction to interfaces ` for a more detailed + explanation of why this is useful. + + :class:`Signature` can be used as a base class to define :ref:`customized ` + signatures and interface objects. + + .. important:: + + :class:`Signature` objects are immutable. Classes inheriting from :class:`Signature` must + ensure this remains the case when additional functionality is added. + """ + def __init__(self, members: Mapping[str, Member]) -> None: + ... + + def flip(self) -> FlippedSignature[Self]: + """Flip the data flow of the members in this signature. + + Returns + ------- + :class:`FlippedSignature` + Proxy object :pc:`FlippedSignature(self)` that flips the data flow of the attributes + corresponding to the members that are accessed using it. + + See the documentation for the :class:`FlippedSignature` class for a detailed discussion + of how this proxy object works. + """ + ... + + @property + def members(self) -> SignatureMembers: + """Members in this signature. + + Returns + ------- + :class:`SignatureMembers` + """ + ... + + def __eq__(self, other) -> bool: + """Compare this signature with another. + + The behavior of this operator depends on the types of the arguments. If both :pc:`self` + and :pc:`other` are instances of the base :class:`Signature` class, they are compared + structurally (the result is :pc:`self.members == other.members`); otherwise they are + compared by identity (the result is :pc:`self is other`). + + Subclasses of :class:`Signature` are expected to override this method to take into account + the specifics of the domain. If the subclass has additional properties that do not influence + the :attr:`members` dictionary but nevertheless make its instance incompatible with other + instances (for example, whether the feedback is combinational or registered), + the overridden method must take that into account. + + Returns + ------- + :class:`bool` + """ + ... + + def flatten(self, obj) -> Iterator[tuple[tuple[str | int, ...], Flow, ValueLike]]: + """Recursively iterate through this signature, retrieving member values from an interface + object. + + .. note:: + + The :ref:`paths ` returned by this method and by + :meth:`SignatureMembers.flatten` differ. This method yield several results for each + :class:`Member` in the collection that has a dimension: + + .. doctest:: + :options: +NORMALIZE_WHITESPACE + + >>> sig = wiring.Signature({ + ... "items": In(1).array(2) + ... }) + >>> obj = sig.create() + >>> list(sig.flatten(obj)) + [(('items', 0), In(1), (sig obj__items__0)), + (('items', 1), In(1), (sig obj__items__1))] + + The :meth:`SignatureMembers.flatten` method yields one result for such a member; see + the documentation for that method for an example. + + Returns + ------- + iterator of (:class:`tuple` of :class:`str` or :class:`int`, :class:`Flow`, :ref:`value-like `) + Tuples of :ref:`paths `, flow, and the corresponding member values. A path + yielded by this method is a tuple of strings or integers where each item is an attribute + name or index (correspondingly) using which the member value was retrieved. + """ + ... + + def is_compliant(self, obj, *, reasons: Optional[list[str]] =..., path: tuple[str, ...] =...) -> bool: + """Check whether an object matches the description in this signature. + + This module places few restrictions on what an interface object may be; it does not + prescribe a specific base class or a specific way of constructing the object, only + the values that its attributes should have. This method ensures consistency between + the signature and the interface object, checking every aspect of the provided interface + object for compliance with the signature. + + It verifies that: + + * :pc:`obj` has a :pc:`signature` attribute whose value a :class:`Signature` instance + such that ``self == obj.signature``; + * for each member, :pc:`obj` has an attribute with the same name, whose value: + + * for members with :meth:`dimensions ` specified, contains a list or + a tuple (or several levels of nested lists or tuples, for multiple dimensions) + satisfying the requirements below; + * for port members, is a :ref:`value-like ` object casting to + a :class:`Signal` or a :class:`Const` whose width and signedness is the same as that + of the member, and (in case of a :class:`Signal`) which is not reset-less and whose + reset value is that of the member; + * for signature members, matches the description in the signature as verified by + :meth:`Signature.is_compliant`. + + If the verification fails, this method reports the reason(s) by filling the :pc:`reasons` + container. These reasons are intended to be human-readable: more than one reason may be + reported but only in cases where this is helpful (e.g. the same error message will not + repeat 10 times for each of the 10 ports in a list). + + Arguments + --------- + reasons : :class:`list` or :pc:`None` + If provided, a container that receives diagnostic messages. + path : :class:`tuple` of :class:`str` + The :ref:`path ` to :pc:`obj`. Could be set to improve diagnostic + messages if :pc:`obj` is nested within another object, or for clarity. + + Returns + ------- + :class:`bool` + :pc:`True` if :pc:`obj` matches the description in this signature, :pc:`False` + otherwise. If :pc:`False` and :pc:`reasons` was not :pc:`None`, it will contain + a detailed explanation why. + """ + ... + + def create(self, *, path: tuple[str | int, ...]=..., src_loc_at: int =...) -> AbstractInterface[Self]: + """Create an interface object from this signature. + + The default :meth:`Signature.create` implementation consists of one line: + + .. code:: + + def create(self, *, path=None, src_loc_at=0): + return PureInterface(self, path=path, src_loc_at=1 + src_loc_at) + + This implementation creates an interface object from this signature that serves purely + as a container for the attributes corresponding to the signature members, and implements + no behavior. Such an implementation is sufficient for signatures created ad-hoc using + the :pc:`Signature({ ... })` constructor as well as simple signature subclasses. + + When defining a :class:`Signature` subclass that needs to customize the behavior of + the created interface objects, override this method with a similar implementation + that references the class of your custom interface object: + + .. testcode:: + + class CustomSignature(wiring.Signature): + def create(self, *, path=None, src_loc_at=0): + return CustomInterface(self, path=path, src_loc_at=1 + src_loc_at) + + class CustomInterface(wiring.PureInterface): + @property + def my_property(self): + ... + + The :pc:`path` and :pc:`src_loc_at` arguments are necessary to ensure the generated signals + have informative names and accurate source location information. + + The custom :meth:`create` method may take positional or keyword arguments in addition to + the two listed above. Such arguments must have a default value, because + the :meth:`SignatureMembers.create` method will call the :meth:`Signature.create` member + without these additional arguments when this signature is a member of another signature. + """ + ... + + def __repr__(self) -> str: + ... + + + +@final +class FlippedSignature(Generic[_T_Signature]): + """Description of an interface object, with the members' directions flipped. + + Although an instance of :class:`FlippedSignature` could be created directly, it will be usually + created by a call to :meth:`Signature.flip`. + + This proxy is a wrapper around :class:`Signature` that contains the same description as + the inner mapping, but flips the members' data flow when they are accessed. It is useful + because :class:`Signature` objects are mutable and may include custom behavior, and if one was + copied (rather than wrapped) by :meth:`Signature.flip`, the wrong object would be mutated, and + custom behavior would be unavailable. + + For example: + + .. testcode:: + + sig = wiring.Signature({"foo": Out(1)}) + + flipped_sig = sig.flip() + assert flipped_sig.members["foo"].flow == In + + sig.attr = 1 + assert flipped_sig.attr == 1 + flipped_sig.attr += 1 + assert sig.attr == flipped_sig.attr == 2 + + This class implements the same methods, with the same functionality (other than the flipping of + the members' data flow), as the :class:`Signature` class; see the documentation for that class + for details. + + It is not possible to inherit from :class:`FlippedSignature` and :meth:`Signature.flip` must not + be overridden. If a :class:`Signature` subclass defines a method and this method is called on + a flipped instance of the subclass, it receives the flipped instance as its :pc:`self` argument. + To distinguish being called on the flipped instance from being called on the unflipped one, use + :pc:`isinstance(self, FlippedSignature)`: + + .. testcode:: + + class SignatureKnowsWhenFlipped(wiring.Signature): + @property + def is_flipped(self): + return isinstance(self, wiring.FlippedSignature) + + sig = SignatureKnowsWhenFlipped({}) + assert sig.is_flipped == False + assert sig.flip().is_flipped == True + """ + def __init__(self, signature: _T_Signature) -> None: + ... + + def flip(self) -> _T_Signature: + """ + Flips this signature back to the original one. + + Returns + ------- + :class:`Signature` + :pc:`unflipped` + """ + ... + + @property + def members(self) -> FlippedSignatureMembers: + ... + + def __eq__(self, other) -> bool: + ... + + def flatten(self, obj) -> Iterator[tuple[tuple[str | int, ...], Flow, ValueLike]]: + ... + + def is_compliant(self, obj, *, reasons: Optional[list[str]] =..., path: tuple[str, ...] =...) -> bool: + ... + + def __getattr__(self, name) -> Any: + """Retrieves attribute or method :pc:`name` of the unflipped signature. + + Performs :pc:`getattr(unflipped, name)`, ensuring that, if :pc:`name` refers to a property + getter or a method, its :pc:`self` argument receives the *flipped* signature. A class + method's :pc:`cls` argument receives the class of the *unflipped* signature, as usual. + """ + ... + + def __setattr__(self, name, value) -> None: + """Assigns attribute :pc:`name` of the unflipped signature to ``value``. + + Performs :pc:`setattr(unflipped, name, value)`, ensuring that, if :pc:`name` refers to + a property setter, its :pc:`self` argument receives the flipped signature. + """ + ... + + def __delattr__(self, name) -> None: + """Removes attribute :pc:`name` of the unflipped signature. + + Performs :pc:`delattr(unflipped, name)`, ensuring that, if :pc:`name` refers to a property + deleter, its :pc:`self` argument receives the flipped signature. + """ + ... + + def create(self, *args, path: tuple[str | int, ...] =..., src_loc_at: int =..., **kwargs) -> FlippedInterface: + ... + + def __repr__(self) -> str: + ... + + + +class PureInterface(Generic[_T_Signature]): + """A helper for constructing ad-hoc interfaces. + + The :class:`PureInterface` helper primarily exists to be used by the default implementation of + :meth:`Signature.create`, but it can also be used in any other context where an interface + object needs to be created without the overhead of defining a class for it. + + .. important:: + + Any object can be an interface object; it only needs a :pc:`signature` property containing + a compliant signature. It is **not** necessary to use :class:`PureInterface` in order to + create an interface object, but it may be used either directly or as a base class whenever + it is convenient to do so. + """ + signature: _T_Signature + + def __init__(self, signature: _T_Signature, *, path: tuple[str | int, ...]=..., src_loc_at: int =...) -> None: + """Create attributes from a signature. + + The sole method defined by this helper is its constructor, which only defines + the :pc:`self.signature` attribute as well as the attributes created from the signature + members: + + .. code:: + + def __init__(self, signature, *, path): + self.__dict__.update({ + "signature": signature, + **signature.members.create(path=path) + }) + + .. note:: + + This implementation can be copied and reused in interface objects that *do* include + custom behavior, if the signature serves as the source of truth for attributes + corresponding to its members. Although it is less repetitive, this approach can confuse + IDEs and type checkers. + """ + ... + + def __repr__(self) -> str: + ... + + + +@final +class FlippedInterface(Generic[_T_Signature, _T_Interface]): + """An interface object, with its members' directions flipped. + + An instance of :class:`FlippedInterface` should only be created by calling :func:`flipped`, + which ensures that a :pc:`FlippedInterface(FlippedInterface(...))` object is never created. + + This proxy wraps any interface object and forwards attribute and method access to the wrapped + interface object while flipping its signature and the values of any attributes corresponding to + interface members. It is useful because interface objects may be mutable or include custom + behavior, and explicitly keeping track of whether the interface object is flipped would be very + burdensome. + + For example: + + .. testcode:: + + intf = wiring.PureInterface(wiring.Signature({"foo": Out(1)}), path=()) + + flipped_intf = wiring.flipped(intf) + assert flipped_intf.signature.members["foo"].flow == In + + intf.attr = 1 + assert flipped_intf.attr == 1 + flipped_intf.attr += 1 + assert intf.attr == flipped_intf.attr == 2 + + It is not possible to inherit from :class:`FlippedInterface`. If an interface object class + defines a method or a property and it is called on the flipped interface object, the method + receives the flipped interface object as its :pc:`self` argument. To distinguish being called + on the flipped interface object from being called on the unflipped one, use + :pc:`isinstance(self, FlippedInterface)`: + + .. testcode:: + + class InterfaceKnowsWhenFlipped: + signature = wiring.Signature({}) + + @property + def is_flipped(self): + return isinstance(self, wiring.FlippedInterface) + + intf = InterfaceKnowsWhenFlipped() + assert intf.is_flipped == False + assert wiring.flipped(intf).is_flipped == True + """ + def __init__(self, interface: _T_Interface) -> None: + ... + + # not true -- this is a property -- but required for clean typing + signature: _T_Signature +# @property +# def signature(self) -> _T_Signature: +# """Signature of the flipped interface. +# +# Returns +# ------- +# Signature +# :pc:`unflipped.signature.flip()` +# """ +# ... + + def __eq__(self, other) -> bool: + """Compare this flipped interface with another. + + Returns + ------- + bool + :pc:`True` if :pc:`other` is an instance :pc:`FlippedInterface(other_unflipped)` where + :pc:`unflipped == other_unflipped`, :pc:`False` otherwise. + """ + ... + + def __getattr__(self, name) -> Any: + """Retrieves attribute or method :pc:`name` of the unflipped interface. + + Performs :pc:`getattr(unflipped, name)`, with the following caveats: + + 1. If :pc:`name` refers to a signature member, the returned interface object is flipped. + 2. If :pc:`name` refers to a property getter or a method, its :pc:`self` argument receives + the *flipped* interface. A class method's :pc:`cls` argument receives the class of + the *unflipped* interface, as usual. + """ + ... + + def __setattr__(self, name, value) -> None: + """Assigns attribute :pc:`name` of the unflipped interface to ``value``. + + Performs :pc:`setattr(unflipped, name, value)`, with the following caveats: + + 1. If :pc:`name` refers to a signature member, the assigned interface object is flipped. + 2. If :pc:`name` refers to a property setter, its :pc:`self` argument receives the flipped + interface. + """ + ... + + def __delattr__(self, name) -> None: + """Removes attribute :pc:`name` of the unflipped interface. + + Performs :pc:`delattr(unflipped, name)`, ensuring that, if :pc:`name` refers to a property + deleter, its :pc:`self` argument receives the flipped interface. + """ + ... + + def __repr__(self) -> str: + ... + +@overload +def flipped(interface: FlippedInterface[_T_Signature, _T_Interface]) -> _T_Interface: + ... + +# Can't be typed nicer for now. +@overload +def flipped(interface: _T_Interface) -> FlippedInterface[Any, _T_Interface]: + ... + +def flipped(interface: _T_Interface) -> _T_Interface | FlippedInterface[Any, _T_Interface]: + """ + Flip the data flow of the members of the interface object :pc:`interface`. + + If an interface object is flipped twice, returns the original object: + :pc:`flipped(flipped(interface)) is interface`. Otherwise, wraps :pc:`interface` in + a :class:`FlippedInterface` proxy object that flips the directions of its members. + + See the documentation for the :class:`FlippedInterface` class for a detailed discussion of how + this proxy object works. + """ + ... + +@final +class ConnectionError(Exception): + """Exception raised when the :func:`connect` function is requested to perform an impossible, + meaningless, or forbidden connection.""" + ... + + +def connect(m: ModuleLike, *args: AbstractInterface, **kwargs: AbstractInterface) -> None: + """Connect interface objects to each other. + + This function creates connections between ports of several interface objects. (Any number of + interface objects may be provided; in most cases it is two.) + + The connections can be made only if all of the objects satisfy a number of requirements: + + * Every interface object must have the same set of port members, and they must have the same + :meth:`dimensions `. + * For each path, the port members of every interface object must have the same width and reset + value (for port members corresponding to signals) or constant value (for port members + corresponding to constants). Signedness may differ. + * For each path, at most one interface object must have the corresponding port member be + an output. + * For a given path, if any of the interface objects has an input port member corresponding + to a constant value, then the rest of the interface objects must have output port members + corresponding to the same constant value. + + For example, if :pc:`obj1` is being connected to :pc:`obj2` and :pc:`obj3`, and :pc:`obj1.a.b` + is an output, then :pc:`obj2.a.b` and :pc:`obj2.a.b` must exist and be inputs. If :pc:`obj2.c` + is an input and its value is :pc:`Const(1)`, then :pc:`obj1.c` and :pc:`obj3.c` must be outputs + whose value is also :pc:`Const(1)`. If no ports besides :pc:`obj1.a.b` and :pc:`obj1.c` exist, + then no ports except for those two must exist on :pc:`obj2` and :pc:`obj3` either. + + Once it is determined that the interface objects can be connected, this function performs + an equivalent of: + + .. code:: + + m.d.comb += [ + in1.eq(out1), + in2.eq(out1), + ... + ] + + Where :pc:`out1` is an output and :pc:`in1`, :pc:`in2`, ... are the inputs that have the same + path. (If no interface object has an output for a given path, **no connection at all** is made.) + + The positions (within :pc:`args`) or names (within :pc:`kwargs`) of the arguments do not affect + the connections that are made. There is no difference in behavior between :pc:`connect(m, a, b)` + and :pc:`connect(m, b, a)` or :pc:`connect(m, arbiter=a, decoder=b)`. The names of the keyword + arguments serve only a documentation purpose: they clarify the diagnostic messages when + a connection cannot be made. + """ + ... + +class Component(Elaboratable, Generic[_T_Signature]): + """Base class for elaboratable interface objects. + + A component is an :class:`Elaboratable` whose interaction with other parts of the design is + defined by its signature. Most if not all elaboratables in idiomatic Amaranth code should be + components, as the signature clarifies the direction of data flow at their boundary. See + the :ref:`introduction to interfaces ` section for a practical guide to defining + and using components. + + There are two ways to define a component. If all instances of a component have the same + signature, it can be defined using :term:`variable annotations `: + + .. testcode:: + + class FixedComponent(wiring.Component): + en: In(1) + data: Out(8) + + The variable annotations are collected by the constructor :meth:`Component.__init__`. Only + public (not starting with ``_``) annotations with :class:`In ` or :class:`Out ` + objects are considered; all other annotations are ignored under the assumption that they are + interpreted by some other tool. + + It is possible to use inheritance to extend a component: the component's signature is composed + from the variable annotations in the class that is being constructed as well as all of its + base classes. It is an error to have more than one variable annotation for the same attribute. + + If different instances of a component may need to have different signatures, variable + annotations cannot be used. In this case, the constructor should be overridden, and + the computed signature members should be provided to the superclass constructor: + + .. testcode:: + + class ParametricComponent(wiring.Component): + def __init__(self, data_width): + super().__init__({ + "en": In(1), + "data": Out(data_width) + }) + + It is also possible to pass a :class:`Signature` instance to the superclass constructor. + + Aside from initializing the :attr:`signature` attribute, the :meth:`Component.__init__` + constructor creates attributes corresponding to all of the members defined in the signature. + If an attribute with the same name as that of a member already exists, an error is raied. + + Raises + ------ + :exc:`TypeError` + If the :pc:`signature` object is neither a :class:`Signature` nor a :class:`dict`. + If neither variable annotations nor the :pc:`signature` argument are present, or if + both are present. + :exc:`NameError` + If a name conflict is detected between two variable annotations, or between a member + and an existing attribute. + """ + def __init__(self, signature: Optional[_T_Signature | dict[str, Member]] = None) -> None: + ... + + @property + def signature(self) -> _T_Signature: + """The signature of the component. + + .. important:: + + Do not override this property. Once a component is constructed, its :attr:`signature` + property must always return the same :class:`Signature` instance. The constructor + can be used to customize a component's signature. + """ + ... + + + diff --git a/test/peripherals/test_axi_lite.py b/test/peripherals/test_axi_lite.py index e1d52c107..27821c899 100644 --- a/test/peripherals/test_axi_lite.py +++ b/test/peripherals/test_axi_lite.py @@ -6,7 +6,7 @@ class AXILiteInterfaceWrapper: - def __init__(self, axi_lite_master: Record): + def __init__(self, axi_lite_master: AXILiteInterface): self.axi_lite = axi_lite_master def slave_ra_ready(self, rdy=1): diff --git a/test/peripherals/test_wishbone.py b/test/peripherals/test_wishbone.py index a8332f62b..4dd5485ed 100644 --- a/test/peripherals/test_wishbone.py +++ b/test/peripherals/test_wishbone.py @@ -1,6 +1,8 @@ import random from collections import deque +from amaranth.lib.wiring import connect + from coreblocks.peripherals.wishbone import * from transactron.lib import AdapterTrans @@ -9,8 +11,8 @@ class WishboneInterfaceWrapper: - def __init__(self, wishbone_record): - self.wb = wishbone_record + def __init__(self, wishbone_interface: WishboneInterface): + self.wb = wishbone_interface def master_set(self, addr, data, we): yield self.wb.dat_w.eq(data) @@ -142,10 +144,10 @@ def slave(): class TestWishboneMuxer(TestCaseWithSimulator): def test_manual(self): - wb_master = WishboneInterfaceWrapper(Record(WishboneLayout(WishboneParameters()).wb_layout)) num_slaves = 4 - slaves = [WishboneInterfaceWrapper(Record.like(wb_master.wb, name=f"sl{i}")) for i in range(num_slaves)] - mux = WishboneMuxer(wb_master.wb, [s.wb for s in slaves], Signal(num_slaves)) + mux = WishboneMuxer(WishboneParameters(), num_slaves, Signal(num_slaves)) + slaves = [WishboneInterfaceWrapper(slave) for slave in mux.slaves] + wb_master = WishboneInterfaceWrapper(mux.master_wb) def process(): # check full communiaction @@ -183,9 +185,9 @@ def process(): class TestWishboneAribiter(TestCaseWithSimulator): def test_manual(self): - slave = WishboneInterfaceWrapper(Record(WishboneLayout(WishboneParameters()).wb_layout)) - masters = [WishboneInterfaceWrapper(Record.like(slave.wb, name=f"mst{i}")) for i in range(2)] - arb = WishboneArbiter(slave.wb, [m.wb for m in masters]) + arb = WishboneArbiter(WishboneParameters(), 2) + slave = WishboneInterfaceWrapper(arb.slave_wb) + masters = [WishboneInterfaceWrapper(master) for master in arb.masters] def process(): yield from masters[0].master_set(2, 3, 1) @@ -319,7 +321,7 @@ def elaborate(self, platform): m.submodules.request = self.request = TestbenchIO(AdapterTrans(self.mem_master.request)) m.submodules.result = self.result = TestbenchIO(AdapterTrans(self.mem_master.result)) - m.d.comb += self.mem_master.wb_master.connect(self.mem_slave.bus) + connect(m, self.mem_master.wb_master, self.mem_slave.bus) return m diff --git a/test/regression/pysim.py b/test/regression/pysim.py index 424d83d8e..7eccf3c9d 100644 --- a/test/regression/pysim.py +++ b/test/regression/pysim.py @@ -18,7 +18,7 @@ from coreblocks.core import Core from coreblocks.params import GenParams from coreblocks.params.configurations import full_core_config -from coreblocks.peripherals.wishbone import WishboneBus +from coreblocks.peripherals.wishbone import WishboneSignature class PySimulation(SimulationBackend): @@ -131,8 +131,8 @@ def pretty_dump_metrics(self, metric_values: dict[str, dict[str, int]], filter_r async def run(self, mem_model: CoreMemoryModel, timeout_cycles: int = 5000) -> SimulationExecutionResult: with DependencyContext(DependencyManager()): - wb_instr_bus = WishboneBus(self.gp.wb_params) - wb_data_bus = WishboneBus(self.gp.wb_params) + wb_instr_bus = WishboneSignature(self.gp.wb_params).create() + wb_data_bus = WishboneSignature(self.gp.wb_params).create() core = Core(gen_params=self.gp, wb_instr_bus=wb_instr_bus, wb_data_bus=wb_data_bus) wb_instr_ctrl = WishboneInterfaceWrapper(wb_instr_bus) diff --git a/test/test_core.py b/test/test_core.py index 8bf5c8f1b..a2cfd1d88 100644 --- a/test/test_core.py +++ b/test/test_core.py @@ -1,4 +1,5 @@ from amaranth import Elaboratable, Module +from amaranth.lib.wiring import connect from transactron.lib import AdapterTrans from transactron.utils import align_to_power_of_two, signed_to_int @@ -8,7 +9,7 @@ from coreblocks.core import Core from coreblocks.params import GenParams from coreblocks.params.configurations import CoreConfiguration, basic_core_config, full_core_config -from coreblocks.peripherals.wishbone import WishboneBus, WishboneMemorySlave +from coreblocks.peripherals.wishbone import WishboneSignature, WishboneMemorySlave from typing import Optional import random @@ -33,8 +34,8 @@ def __init__(self, gen_params: GenParams, instr_mem: list[int] = [0], data_mem: def elaborate(self, platform): m = Module() - wb_instr_bus = WishboneBus(self.gen_params.wb_params) - wb_data_bus = WishboneBus(self.gen_params.wb_params) + wb_instr_bus = WishboneSignature(self.gen_params.wb_params).create() + wb_data_bus = WishboneSignature(self.gen_params.wb_params).create() # Align the size of the memory to the length of a cache line. instr_mem_depth = align_to_power_of_two(len(self.instr_mem), self.gen_params.icache_params.block_size_bits) @@ -54,8 +55,8 @@ def elaborate(self, platform): m.submodules.io_in = self.io_in m.submodules.interrupt = self.interrupt - m.d.comb += wb_instr_bus.connect(self.wb_mem_slave.bus) - m.d.comb += wb_data_bus.connect(self.wb_mem_slave_data.bus) + connect(m, wb_instr_bus, self.wb_mem_slave.bus) + connect(m, wb_data_bus, self.wb_mem_slave_data.bus) return m diff --git a/transactron/utils/_typing.py b/transactron/utils/_typing.py index 8f42c1910..a44dc6610 100644 --- a/transactron/utils/_typing.py +++ b/transactron/utils/_typing.py @@ -14,11 +14,12 @@ Any, TYPE_CHECKING, ) -from collections.abc import Iterable, Mapping, Sequence +from collections.abc import Iterable, Iterator, Mapping, Sequence from contextlib import AbstractContextManager from enum import Enum from amaranth import * from amaranth.lib.data import StructLayout, View +from amaranth.lib.wiring import Flow, Member from amaranth.hdl import ShapeCastable, ValueCastable from amaranth.hdl.rec import Direction, Layout @@ -86,6 +87,7 @@ GraphCC: TypeAlias = set[T] +# Protocols for Amaranth classes class _ModuleBuilderDomainsLike(Protocol): def __getattr__(self, name: str) -> "_ModuleBuilderDomain": ... @@ -143,6 +145,74 @@ def next(self, name: str) -> None: ... +class AbstractSignatureMembers(Protocol): + def flip(self) -> "AbstractSignatureMembers": + ... + + def __eq__(self, other) -> bool: + ... + + def __contains__(self, name: str) -> bool: + ... + + def __getitem__(self, name: str) -> Member: + ... + + def __setitem__(self, name: str, member: Member) -> NoReturn: + ... + + def __delitem__(self, name: str) -> NoReturn: + ... + + def __iter__(self) -> Iterator[str]: + ... + + def __len__(self) -> int: + ... + + def flatten(self, *, path: tuple[str | int, ...] = ...) -> Iterator[tuple[tuple[str | int, ...], Member]]: + ... + + def create(self, *, path: tuple[str | int, ...] = ..., src_loc_at: int = ...) -> dict[str, Any]: + ... + + def __repr__(self) -> str: + ... + + +class AbstractSignature(Protocol): + def flip(self) -> "AbstractSignature": + ... + + @property + def members(self) -> AbstractSignatureMembers: + ... + + def __eq__(self, other) -> bool: + ... + + def flatten(self, obj) -> Iterator[tuple[tuple[str | int, ...], Flow, ValueLike]]: + ... + + def is_compliant(self, obj, *, reasons: Optional[list[str]] = ..., path: tuple[str, ...] = ...) -> bool: + ... + + def create( + self, *, path: tuple[str | int, ...] = ..., src_loc_at: int = ... + ) -> "AbstractInterface[AbstractSignature]": + ... + + def __repr__(self) -> str: + ... + + +_T_AbstractSignature = TypeVar("_T_AbstractSignature", bound=AbstractSignature) + + +class AbstractInterface(Protocol, Generic[_T_AbstractSignature]): + signature: _T_AbstractSignature + + class HasElaborate(Protocol): def elaborate(self, platform) -> "HasElaborate": ... From 78cb3f5a5b2fe915b11c460feae60c588f2b27bd Mon Sep 17 00:00:00 2001 From: Marek Materzok Date: Thu, 14 Mar 2024 10:19:58 +0100 Subject: [PATCH 5/7] Destroy all records (#612) --- coreblocks/cache/icache.py | 28 ++++++++++++++------------ coreblocks/fu/div_unit.py | 3 ++- coreblocks/fu/fu_decoder.py | 2 +- coreblocks/lsu/pma.py | 15 +++++++++----- coreblocks/params/layouts.py | 6 ------ coreblocks/structs_common/rf.py | 5 +++-- stubs/amaranth/hdl/_ast.pyi | 6 +++--- test/transactions/test_assign.py | 16 +++++++-------- transactron/lib/storage.py | 17 ++++++++-------- transactron/utils/_typing.py | 7 +------ transactron/utils/assign.py | 34 ++++++++++---------------------- 11 files changed, 62 insertions(+), 77 deletions(-) diff --git a/coreblocks/cache/icache.py b/coreblocks/cache/icache.py index 7b54b9675..09899afb6 100644 --- a/coreblocks/cache/icache.py +++ b/coreblocks/cache/icache.py @@ -2,6 +2,7 @@ import operator from amaranth import * +from amaranth.lib.data import View from amaranth.utils import exact_log2 from transactron.core import def_method, Priority, TModule @@ -12,6 +13,7 @@ from coreblocks.peripherals.bus_adapter import BusMasterInterface from coreblocks.cache.iface import CacheInterface, CacheRefillerInterface +from transactron.utils.transactron_helpers import make_layout __all__ = [ "ICache", @@ -109,11 +111,11 @@ def __init__(self, layouts: ICacheLayouts, params: ICacheParameters, refiller: C self.flush = Method() self.flush.add_conflict(self.issue_req, Priority.LEFT) - self.addr_layout = [ + self.addr_layout = make_layout( ("offset", self.params.offset_bits), ("index", self.params.index_bits), ("tag", self.params.tag_bits), - ] + ) self.perf_loads = HwCounter("frontend.icache.loads", "Number of requests to the L1 Instruction Cache") self.perf_hits = HwCounter("frontend.icache.hits") @@ -131,7 +133,7 @@ def deserialize_addr(self, raw_addr: Value) -> dict[str, Value]: "tag": raw_addr[-self.params.tag_bits :], } - def serialize_addr(self, addr: Record) -> Value: + def serialize_addr(self, addr: View) -> Value: return Cat(addr.offset, addr.index, addr.tag) def elaborate(self, platform): @@ -186,7 +188,7 @@ def elaborate(self, platform): # Fast path - read requests request_valid = self.req_fifo.read.ready - request_addr = Record(self.addr_layout) + request_addr = Signal(self.addr_layout) tag_hit = [tag_data.valid & (tag_data.tag == request_addr.tag) for tag_data in self.mem.tag_rd_data] tag_hit_any = reduce(operator.or_, tag_hit) @@ -195,7 +197,7 @@ def elaborate(self, platform): for i in OneHotSwitchDynamic(m, Cat(tag_hit)): m.d.comb += mem_out.eq(self.mem.data_rd_data[i]) - instr_out = extract_instr_from_word(m, self.params, mem_out, request_addr[:]) + instr_out = extract_instr_from_word(m, self.params, mem_out, Value.cast(request_addr)) refill_error_saved = Signal() m.d.comb += needs_refill.eq(request_valid & ~tag_hit_any & ~refill_error_saved) @@ -214,7 +216,7 @@ def _(): self.req_latency.stop(m) return self.res_fwd.read(m) - mem_read_addr = Record(self.addr_layout) + mem_read_addr = Signal(self.addr_layout) m.d.comb += assign(mem_read_addr, request_addr) @def_method(m, self.issue_req, ready=accepting_requests) @@ -304,21 +306,21 @@ class ICacheMemory(Elaboratable): def __init__(self, params: ICacheParameters) -> None: self.params = params - self.tag_data_layout = [("valid", 1), ("tag", self.params.tag_bits)] + self.tag_data_layout = make_layout(("valid", 1), ("tag", self.params.tag_bits)) self.way_wr_en = Signal(self.params.num_of_ways) self.tag_rd_index = Signal(self.params.index_bits) - self.tag_rd_data = Array([Record(self.tag_data_layout) for _ in range(self.params.num_of_ways)]) + self.tag_rd_data = Array([Signal(self.tag_data_layout) for _ in range(self.params.num_of_ways)]) self.tag_wr_index = Signal(self.params.index_bits) self.tag_wr_en = Signal() - self.tag_wr_data = Record(self.tag_data_layout) + self.tag_wr_data = Signal(self.tag_data_layout) - self.data_addr_layout = [("index", self.params.index_bits), ("offset", self.params.offset_bits)] + self.data_addr_layout = make_layout(("index", self.params.index_bits), ("offset", self.params.offset_bits)) - self.data_rd_addr = Record(self.data_addr_layout) + self.data_rd_addr = Signal(self.data_addr_layout) self.data_rd_data = Array([Signal(self.params.word_width) for _ in range(self.params.num_of_ways)]) - self.data_wr_addr = Record(self.data_addr_layout) + self.data_wr_addr = Signal(self.data_addr_layout) self.data_wr_en = Signal() self.data_wr_data = Signal(self.params.word_width) @@ -328,7 +330,7 @@ def elaborate(self, platform): for i in range(self.params.num_of_ways): way_wr = self.way_wr_en[i] - tag_mem = Memory(width=len(self.tag_wr_data), depth=self.params.num_of_sets) + tag_mem = Memory(width=len(Value.cast(self.tag_wr_data)), depth=self.params.num_of_sets) tag_mem_rp = tag_mem.read_port() tag_mem_wp = tag_mem.write_port() m.submodules[f"tag_mem_{i}_rp"] = tag_mem_rp diff --git a/coreblocks/fu/div_unit.py b/coreblocks/fu/div_unit.py index a4767a0b0..9e3f3dfc6 100644 --- a/coreblocks/fu/div_unit.py +++ b/coreblocks/fu/div_unit.py @@ -3,6 +3,7 @@ from collections.abc import Sequence from amaranth import * +from amaranth.lib import data from coreblocks.params.fu_params import FunctionalComponentParams from coreblocks.params import Funct3, GenParams, FuncUnitLayouts, OpType @@ -33,7 +34,7 @@ def get_instructions(self) -> Sequence[tuple]: ] -def get_input(arg: Record) -> tuple[Value, Value]: +def get_input(arg: data.View) -> tuple[Value, Value]: return arg.s1_val, Mux(arg.imm, arg.imm, arg.s2_val) diff --git a/coreblocks/fu/fu_decoder.py b/coreblocks/fu/fu_decoder.py index 30373e677..eeaae8bf1 100644 --- a/coreblocks/fu/fu_decoder.py +++ b/coreblocks/fu/fu_decoder.py @@ -15,7 +15,7 @@ class Decoder(Elaboratable): Attributes ---------- decode_fn: Signal - exec_fn: Record + exec_fn: View """ def __init__(self, gen_params: GenParams, decode_fn: Type[IntFlag], ops: Sequence[tuple], check_optype: bool): diff --git a/coreblocks/lsu/pma.py b/coreblocks/lsu/pma.py index 8e474a6bf..cd91c98f0 100644 --- a/coreblocks/lsu/pma.py +++ b/coreblocks/lsu/pma.py @@ -2,6 +2,7 @@ from functools import reduce from operator import or_ from amaranth import * +from amaranth.lib import data from coreblocks.params import * from transactron.utils import HasElaborate @@ -29,6 +30,11 @@ class PMARegion: mmio: bool = False +class PMALayout(data.StructLayout): + def __init__(self): + super().__init__({"mmio": unsigned(1)}) + + class PMAChecker(Elaboratable): """ Implementation of physical memory attributes checker. It may or may not be a part of LSU. @@ -38,21 +44,20 @@ class PMAChecker(Elaboratable): ---------- addr : Signal Memory address, for which PMAs are requested. - result : Record + result : View PMAs for given address. """ def __init__(self, gen_params: GenParams) -> None: # poor man's interval list self.segments = gen_params.pma - self.attr_layout = gen_params.get(PMALayouts).pma_attrs_layout - self.result = Record(self.attr_layout) + self.result = Signal(PMALayout()) self.addr = Signal(gen_params.isa.xlen) def elaborate(self, platform) -> HasElaborate: m = TModule() - outputs = [Record(self.attr_layout) for _ in self.segments] + outputs = [Signal(PMALayout()) for _ in self.segments] # zero output if addr not in region, propagate value if addr in region for i, segment in enumerate(self.segments): @@ -64,6 +69,6 @@ def elaborate(self, platform) -> HasElaborate: m.d.comb += outputs[i].eq(segment.mmio) # OR all outputs - m.d.comb += self.result.eq(reduce(or_, outputs, 0)) + m.d.comb += self.result.eq(reduce(or_, [Value.cast(o) for o in outputs], 0)) return m diff --git a/coreblocks/params/layouts.py b/coreblocks/params/layouts.py index 066c6dd43..98f69344c 100644 --- a/coreblocks/params/layouts.py +++ b/coreblocks/params/layouts.py @@ -18,7 +18,6 @@ "UnsignedMulUnitLayouts", "RATLayouts", "LSULayouts", - "PMALayouts", "CSRLayouts", "ICacheLayouts", "JumpBranchLayouts", @@ -551,11 +550,6 @@ def __init__(self, gen_params: GenParams): self.accept = make_layout(fields.data, fields.exception, fields.cause) -class PMALayouts: - def __init__(self, gen_params: GenParams): - self.pma_attrs_layout = [("mmio", 1)] - - class CSRLayouts: """Layouts used in the control and status registers.""" diff --git a/coreblocks/structs_common/rf.py b/coreblocks/structs_common/rf.py index 461fab8ed..899e99593 100644 --- a/coreblocks/structs_common/rf.py +++ b/coreblocks/structs_common/rf.py @@ -1,6 +1,7 @@ from amaranth import * from transactron import Method, def_method, TModule from coreblocks.params import RFLayouts, GenParams +from transactron.utils.transactron_helpers import make_layout __all__ = ["RegisterFile"] @@ -9,9 +10,9 @@ class RegisterFile(Elaboratable): def __init__(self, *, gen_params: GenParams): self.gen_params = gen_params layouts = gen_params.get(RFLayouts) - self.internal_layout = [("reg_val", gen_params.isa.xlen), ("valid", 1)] + self.internal_layout = make_layout(("reg_val", gen_params.isa.xlen), ("valid", 1)) self.read_layout = layouts.rf_read_out - self.entries = Array(Record(self.internal_layout) for _ in range(2**gen_params.phys_regs_bits)) + self.entries = Array(Signal(self.internal_layout) for _ in range(2**gen_params.phys_regs_bits)) self.read1 = Method(i=layouts.rf_read_in, o=layouts.rf_read_out) self.read2 = Method(i=layouts.rf_read_in, o=layouts.rf_read_out) diff --git a/stubs/amaranth/hdl/_ast.pyi b/stubs/amaranth/hdl/_ast.pyi index 98c2be9f2..8892c6f6e 100644 --- a/stubs/amaranth/hdl/_ast.pyi +++ b/stubs/amaranth/hdl/_ast.pyi @@ -410,14 +410,14 @@ class Repl(Value): class _SignalMeta(ABCMeta): @overload - def __call__(cls, shape: ShapeCastable[T], src_loc_at = ..., **kwargs) -> T: + def __call__(cls, shape: ShapeCastable[T], src_loc_at: int = ..., **kwargs) -> T: ... @overload - def __call__(cls, shape = ..., src_loc_at = ..., **kwargs) -> Signal: + def __call__(cls, shape: ShapeLike = ..., src_loc_at: int = ..., **kwargs) -> Signal: ... - def __call__(cls, shape = ..., src_loc_at = ..., **kwargs): + def __call__(cls, shape: ShapeLike = ..., src_loc_at: int = ..., **kwargs): ... diff --git a/test/transactions/test_assign.py b/test/transactions/test_assign.py index 47f72800b..73d5b28f9 100644 --- a/test/transactions/test_assign.py +++ b/test/transactions/test_assign.py @@ -3,7 +3,7 @@ from amaranth.lib import data from amaranth.hdl._ast import ArrayProxy, Slice -from transactron.utils._typing import LayoutLike +from transactron.utils._typing import MethodLayout from transactron.utils import AssignType, assign from transactron.utils.assign import AssignArg, AssignFields @@ -54,15 +54,15 @@ def mkstruct(layout): ], ) class TestAssign(TestCase): - # constructs `assign` arguments (records, proxies, dicts) which have an "inner" and "outer" part - # parameterized with a Record-like constructor and a layout of the inner part - build: Callable[[Callable[[LayoutLike], AssignArg], LayoutLike], AssignArg] + # constructs `assign` arguments (views, proxies, dicts) which have an "inner" and "outer" part + # parameterized with a constructor and a layout of the inner part + build: Callable[[Callable[[MethodLayout], AssignArg], MethodLayout], AssignArg] # constructs field specifications for `assign`, takes field specifications for the inner part wrap: Callable[[AssignFields], AssignFields] # extracts the inner part of the structure - extr: Callable[[AssignArg], Record | ArrayProxy] - # Record-like constructor, takes a record layout - mk: Callable[[LayoutLike], AssignArg] + extr: Callable[[AssignArg], ArrayProxy] + # constructor, takes a layout + mk: Callable[[MethodLayout], AssignArg] def test_rhs_exception(self): with self.assertRaises(KeyError): @@ -99,7 +99,7 @@ def test_wrong_bits(self): ("list", layout_ab, layout_ab, ["a", "a"]), ] ) - def test_assign_a(self, name, layout1: LayoutLike, layout2: LayoutLike, atype: AssignType): + def test_assign_a(self, name, layout1: MethodLayout, layout2: MethodLayout, atype: AssignType): lhs = self.build(self.mk, layout1) rhs = self.build(self.mk, layout2) alist = list(assign(lhs, rhs, fields=self.wrap(atype))) diff --git a/transactron/lib/storage.py b/transactron/lib/storage.py index 9549527f8..e6d3e5cf5 100644 --- a/transactron/lib/storage.py +++ b/transactron/lib/storage.py @@ -1,7 +1,7 @@ from amaranth import * from amaranth.utils import * -from transactron.utils.transactron_helpers import from_method_layout +from transactron.utils.transactron_helpers import from_method_layout, make_layout from ..core import * from ..utils import SrcLoc, get_src_loc from typing import Optional @@ -23,7 +23,7 @@ class MemoryBank(Elaboratable): The read request method. Accepts an `addr` from which data should be read. Only ready if there is there is a place to buffer response. read_resp: Method - The read response method. Return `data_layout` Record which was saved on `addr` given by last + The read response method. Return `data_layout` View which was saved on `addr` given by last `read_req` method call. Only ready after `read_req` call. write: Method The write method. Accepts `addr` where data should be saved, `data` in form of `data_layout` @@ -33,7 +33,7 @@ class MemoryBank(Elaboratable): def __init__( self, *, - data_layout: MethodLayout, + data_layout: LayoutList, elem_count: int, granularity: Optional[int] = None, safe_writes: bool = True, @@ -58,7 +58,7 @@ def __init__( Alternatively, the source location to use instead of the default. """ self.src_loc = get_src_loc(src_loc) - self.data_layout = data_layout + self.data_layout = make_layout(*data_layout) self.elem_count = elem_count self.granularity = granularity self.width = from_method_layout(self.data_layout).size @@ -66,9 +66,10 @@ def __init__( self.safe_writes = safe_writes self.read_req_layout: LayoutList = [("addr", self.addr_width)] - self.write_layout = [("addr", self.addr_width), ("data", self.data_layout)] + write_layout = [("addr", self.addr_width), ("data", self.data_layout)] if self.granularity is not None: - self.write_layout.append(("mask", self.width // self.granularity)) + write_layout.append(("mask", self.width // self.granularity)) + self.write_layout = make_layout(*write_layout) self.read_req = Method(i=self.read_req_layout, src_loc=self.src_loc) self.read_resp = Method(o=self.data_layout, src_loc=self.src_loc) @@ -85,8 +86,8 @@ def elaborate(self, platform) -> TModule: prev_read_addr = Signal(self.addr_width) write_pending = Signal() write_req = Signal() - write_args = Record(self.write_layout) - write_args_prev = Record(self.write_layout) + write_args = Signal(self.write_layout) + write_args_prev = Signal(self.write_layout) m.d.comb += read_port.addr.eq(prev_read_addr) zipper = ArgumentsToResultsZipper([("valid", 1)], self.data_layout) diff --git a/transactron/utils/_typing.py b/transactron/utils/_typing.py index a44dc6610..5acd3ab70 100644 --- a/transactron/utils/_typing.py +++ b/transactron/utils/_typing.py @@ -14,14 +14,13 @@ Any, TYPE_CHECKING, ) -from collections.abc import Iterable, Iterator, Mapping, Sequence +from collections.abc import Iterable, Iterator, Mapping from contextlib import AbstractContextManager from enum import Enum from amaranth import * from amaranth.lib.data import StructLayout, View from amaranth.lib.wiring import Flow, Member from amaranth.hdl import ShapeCastable, ValueCastable -from amaranth.hdl.rec import Direction, Layout if TYPE_CHECKING: from amaranth.hdl._ast import Statement @@ -33,7 +32,6 @@ "ValueLike", "ShapeLike", "StatementLike", - "LayoutLike", "SwitchKey", "SrcLoc", "MethodLayout", @@ -60,9 +58,6 @@ ValueLike: TypeAlias = Value | int | Enum | ValueCastable ShapeLike: TypeAlias = Shape | ShapeCastable | int | range | type[Enum] StatementLike: TypeAlias = Union["Statement", Iterable["StatementLike"]] -LayoutLike: TypeAlias = ( - Layout | Sequence[tuple[str, "ShapeLike | LayoutLike"] | tuple[str, "ShapeLike | LayoutLike", Direction]] -) SwitchKey: TypeAlias = str | int | Enum SrcLoc: TypeAlias = tuple[str, int] diff --git a/transactron/utils/assign.py b/transactron/utils/assign.py index b3bee191e..0be471e80 100644 --- a/transactron/utils/assign.py +++ b/transactron/utils/assign.py @@ -41,8 +41,6 @@ def flatten_elems(proxy: ArrayProxy): def assign_arg_fields(val: AssignArg) -> Optional[set[str]]: if isinstance(val, ArrayProxy): return arrayproxy_fields(val) - elif isinstance(val, Record): - return set(val.fields) elif isinstance(val, data.View): layout = val.shape() if isinstance(layout, data.StructLayout): @@ -57,7 +55,7 @@ def assign( """Safe structured assignment. This function recursively generates assignment statements for - field-containing structures. This includes: Amaranth `Record`\\s, + field-containing structures. This includes: Amaranth `View`\\s using `StructLayout`, Python `dict`\\s. In case of mismatching fields or bit widths, error is raised. @@ -68,16 +66,16 @@ def assign( The bit width check is performed if: - - Any of `lhs` or `rhs` is a `Record` or `View`. + - Any of `lhs` or `rhs` is a `View`. - Both `lhs` and `rhs` have an explicitly defined shape (e.g. are a - `Signal`, a field of a `Record` or a `View`). + `Signal`, a field of a `View`). Parameters ---------- - lhs : Record or View or Value-castable or dict - Record, signal or dict being assigned. - rhs : Record or View or Value-castable or dict - Record, signal or dict containing assigned values. + lhs : View or Value-castable or dict + View, signal or dict being assigned. + rhs : View or Value-castable or dict + View, signal or dict containing assigned values. fields : AssignType or Iterable or Mapping, optional Determines which fields will be assigned. Possible values: @@ -109,18 +107,8 @@ def assign( if lhs_fields is not None and rhs_fields is not None: # asserts for type checking - assert ( - isinstance(lhs, Record) - or isinstance(lhs, ArrayProxy) - or isinstance(lhs, Mapping) - or isinstance(lhs, data.View) - ) - assert ( - isinstance(rhs, Record) - or isinstance(rhs, ArrayProxy) - or isinstance(rhs, Mapping) - or isinstance(rhs, data.View) - ) + assert isinstance(lhs, ArrayProxy) or isinstance(lhs, Mapping) or isinstance(lhs, data.View) + assert isinstance(rhs, ArrayProxy) or isinstance(rhs, Mapping) or isinstance(rhs, data.View) if fields is AssignType.COMMON: names = lhs_fields & rhs_fields @@ -166,9 +154,7 @@ def has_explicit_shape(val: ValueLike): return isinstance(val, Signal) or isinstance(val, ArrayProxy) if ( - isinstance(lhs, Record) - or isinstance(rhs, Record) - or isinstance(lhs, data.View) + isinstance(lhs, data.View) or isinstance(rhs, data.View) or (lhs_strict or has_explicit_shape(lhs)) and (rhs_strict or has_explicit_shape(rhs)) From f53c30e3ab7c81041d4e7bdd849f9d27dbaf94df Mon Sep 17 00:00:00 2001 From: piotro888 Date: Thu, 14 Mar 2024 10:47:54 +0100 Subject: [PATCH 6/7] Optimize submodule downloads in CI (#618) --- .github/workflows/main.yml | 86 ++++++++++++++++++++++++++------------ 1 file changed, 59 insertions(+), 27 deletions(-) diff --git a/.github/workflows/main.yml b/.github/workflows/main.yml index 23c1aa56c..437dc2232 100644 --- a/.github/workflows/main.yml +++ b/.github/workflows/main.yml @@ -61,8 +61,14 @@ jobs: steps: - name: Checkout uses: actions/checkout@v3 - with: - submodules: recursive + + - name: Get submodules HEAD hash + working-directory: . + run: | + # ownership workaround + git config --global --add safe.directory /__w/coreblocks/coreblocks + # paths in command are relative! + git submodule > .gitmodules-hash - name: Cache compiled and reference riscv-arch-test id: cache-riscv-arch-test @@ -79,11 +85,17 @@ jobs: '**/test/external/riscof/coreblocks/**', '**/test/external/riscof/spike_simple/**', '**/test/external/riscof/config.ini', - '**/.git/modules/test/external/riscof/riscv-arch-test/HEAD', - '**/docker/riscv-toolchain.Dockerfile' + '**/.gitmodules-hash', + '**/docker/riscv-toolchain.Dockerfile', + '**/.github/workflows/main.yml' ) }} - restore-keys: | - ${{ env.cache-name }}-${{ runner.os }}- + lookup-only: true + + - if: ${{ steps.cache-riscv-arch-test.outputs.cache-hit != 'true' }} + name: Checkout with submodules + uses: actions/checkout@v3 + with: + submodules: recursive - if: ${{ steps.cache-riscv-arch-test.outputs.cache-hit != 'true' }} name: Setup PATH @@ -112,7 +124,9 @@ jobs: name: Build tests for Coreblocks run: | MAKEFILE_PATH=riscof_work/Makefile.build-DUT-coreblocks ../../../ci/riscof_run_makefile.sh + - if: ${{ steps.cache-riscv-arch-test.outputs.cache-hit != 'true' }} + name: Upload compiled and reference tests artifact uses: actions/upload-artifact@v3 with: name: "riscof-tests" @@ -130,8 +144,6 @@ jobs: steps: - name: Checkout uses: actions/checkout@v3 - with: - submodules: recursive - name: Set up Python uses: actions/setup-python@v4 @@ -146,11 +158,18 @@ jobs: python3 -m pip install -r requirements-dev.txt - uses: actions/download-artifact@v3 + name: Download full verilog core with: name: "verilog-full-core" path: . + - name: Get submodules HEAD hash + run: | + git config --global --add safe.directory /__w/coreblocks/coreblocks + git submodule > .gitmodules-hash + - uses: actions/cache@v3 + name: Download tests from cache env: cache-name: cache-riscv-arch-test with: @@ -163,8 +182,9 @@ jobs: '**/test/external/riscof/coreblocks/**', '**/test/external/riscof/spike_simple/**', '**/test/external/riscof/config.ini', - '**/.git/modules/test/external/riscof/riscv-arch-test/HEAD', - '**/docker/riscv-toolchain.Dockerfile' + '**/.gitmodules-hash', + '**/docker/riscv-toolchain.Dockerfile', + '**/.github/workflows/main.yml' ) }} fail-on-cache-miss: true @@ -182,13 +202,14 @@ jobs: runs-on: ubuntu-latest container: ghcr.io/kuznia-rdzeni/riscv-toolchain:2024.03.12 timeout-minutes: 10 - outputs: - cache_hit: ${{ steps.cache-regression.outputs.cache-hit }} steps: - name: Checkout uses: actions/checkout@v3 - with: - submodules: recursive + + - name: Get submodules HEAD hash + run: | + git config --global --add safe.directory /__w/coreblocks/coreblocks + git submodule > .gitmodules-hash - name: Cache regression-tests id: cache-regression @@ -197,15 +218,20 @@ jobs: cache-name: cache-regression-tests with: path: test/external/riscv-tests/test-* - key: ${{ env.cache-name }}-${{ runner.os }}-${{ hashFiles( - '**/test/external/riscv-tests/environment/**', + '**/test/external/riscv-tests/environment/custom/**', '**/test/external/riscv-tests/Makefile', - '**/.git/modules/test/external/riscv-tests/riscv-tests/HEAD', - '**/docker/riscv-toolchain.Dockerfile' + '**/.gitmodules-hash', + '**/docker/riscv-toolchain.Dockerfile', + '**/.github/workflows/main.yml' ) }} - restore-keys: | - ${{ env.cache-name }}-${{ runner.os }}- + lookup-only: true + + - if: ${{ steps.cache-regression.outputs.cache-hit != 'true' }} + name: Checkout with submodules + uses: actions/checkout@v3 + with: + submodules: recursive - if: ${{ steps.cache-regression.outputs.cache-hit != 'true' }} run: cd test/external/riscv-tests && make @@ -225,8 +251,6 @@ jobs: steps: - name: Checkout uses: actions/checkout@v3 - with: - submodules: recursive - name: Set up Python uses: actions/setup-python@v4 @@ -241,21 +265,29 @@ jobs: python3 -m pip install -r requirements-dev.txt - uses: actions/download-artifact@v3 + name: Download full verilog core with: name: "verilog-full-core" path: . + - name: Get submodules HEAD hash + run: | + git config --global --add safe.directory /__w/coreblocks/coreblocks + git submodule > .gitmodules-hash + - uses: actions/cache@v3 + name: Download tests from cache env: cache-name: cache-regression-tests with: path: test/external/riscv-tests/test-* key: ${{ env.cache-name }}-${{ runner.os }}-${{ hashFiles( - '**/test/external/riscv-tests/environment/**', - '**/test/external/riscv-tests/Makefile', - '**/.git/modules/test/external/riscv-tests/riscv-tests/HEAD', - '**/docker/riscv-toolchain.Dockerfile' - ) }} + '**/test/external/riscv-tests/environment/custom/**', + '**/test/external/riscv-tests/Makefile', + '**/.gitmodules-hash', + '**/docker/riscv-toolchain.Dockerfile', + '**/.github/workflows/main.yml' + ) }} fail-on-cache-miss: true - name: Run tests From 1c273ffbe968dbee3176b4e8bdaac20503c1c331 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jakub=20Urba=C5=84czyk?= Date: Thu, 14 Mar 2024 11:28:03 +0000 Subject: [PATCH 7/7] Hardware Logging (#595) --- .github/workflows/main.yml | 2 +- coreblocks/core.py | 4 +- coreblocks/fu/jumpbranch.py | 13 +- coreblocks/peripherals/wishbone.py | 33 ++- pytest.ini | 1 + scripts/run_benchmarks.py | 20 +- scripts/run_signature.py | 18 +- test/conftest.py | 7 + test/regression/cocotb.py | 56 +++-- .../regression/cocotb/benchmark_entrypoint.py | 2 - .../regression/cocotb/signature_entrypoint.py | 2 - test/regression/cocotb/test_entrypoint.py | 2 - test/regression/pysim.py | 46 ++-- test/regression/test_regression.py | 2 +- test/transactron/testing/test_assertion.py | 32 --- test/transactron/testing/test_log.py | 124 ++++++++++ transactron/lib/logging.py | 229 ++++++++++++++++++ transactron/testing/__init__.py | 2 +- transactron/testing/assertion.py | 20 -- transactron/testing/infrastructure.py | 9 +- transactron/testing/logging.py | 108 +++++++++ transactron/utils/__init__.py | 1 - transactron/utils/assertion.py | 60 ----- transactron/utils/gen.py | 71 +++--- 24 files changed, 651 insertions(+), 213 deletions(-) delete mode 100644 test/transactron/testing/test_assertion.py create mode 100644 test/transactron/testing/test_log.py create mode 100644 transactron/lib/logging.py delete mode 100644 transactron/testing/assertion.py create mode 100644 transactron/testing/logging.py delete mode 100644 transactron/utils/assertion.py diff --git a/.github/workflows/main.yml b/.github/workflows/main.yml index 437dc2232..06ceb129d 100644 --- a/.github/workflows/main.yml +++ b/.github/workflows/main.yml @@ -325,7 +325,7 @@ jobs: sudo apt-get install -y binutils-riscv64-unknown-elf - name: Run tests - run: ./scripts/run_tests.py --verbose + run: ./scripts/run_tests.py -v - name: Check traces and profiles run: ./scripts/run_tests.py -t -p -c 1 TestCore diff --git a/coreblocks/core.py b/coreblocks/core.py index e52ad9f2c..a91b2e827 100644 --- a/coreblocks/core.py +++ b/coreblocks/core.py @@ -49,8 +49,8 @@ def __init__(self, *, gen_params: GenParams, wb_instr_bus: WishboneInterface, wb self.wb_instr_bus = wb_instr_bus self.wb_data_bus = wb_data_bus - self.wb_master_instr = WishboneMaster(self.gen_params.wb_params) - self.wb_master_data = WishboneMaster(self.gen_params.wb_params) + self.wb_master_instr = WishboneMaster(self.gen_params.wb_params, "instr") + self.wb_master_data = WishboneMaster(self.gen_params.wb_params, "data") self.bus_master_instr_adapter = WishboneMasterAdapter(self.wb_master_instr) self.bus_master_data_adapter = WishboneMasterAdapter(self.wb_master_data) diff --git a/coreblocks/fu/jumpbranch.py b/coreblocks/fu/jumpbranch.py index ae2014e7b..8b4ba52c9 100644 --- a/coreblocks/fu/jumpbranch.py +++ b/coreblocks/fu/jumpbranch.py @@ -7,8 +7,8 @@ from transactron import * from transactron.core import def_method from transactron.lib import * +from transactron.lib import logging from transactron.utils import DependencyManager - from coreblocks.params import * from coreblocks.params.keys import AsyncInterruptInsertSignalKey, BranchVerifyKey from transactron.utils import OneHotSwitch @@ -19,6 +19,9 @@ __all__ = ["JumpBranchFuncUnit", "JumpComponent"] +log = logging.HardwareLogger("backend.fu.jumpbranch") + + class JumpBranchFn(DecoderManager): class Fn(IntFlag): JAL = auto() @@ -209,6 +212,14 @@ def _(arg): with m.If(~is_auipc): self.fifo_branch_resolved.write(m, from_pc=jb.in_pc, next_pc=jump_result, misprediction=misprediction) + log.debug( + m, + True, + "jumping from 0x{:08x} to 0x{:08x}; misprediction: {}", + jb.in_pc, + jump_result, + misprediction, + ) return m diff --git a/coreblocks/peripherals/wishbone.py b/coreblocks/peripherals/wishbone.py index e71617682..f2dcca253 100644 --- a/coreblocks/peripherals/wishbone.py +++ b/coreblocks/peripherals/wishbone.py @@ -11,6 +11,7 @@ from transactron.utils._typing import AbstractInterface, AbstractSignature from transactron.lib.connectors import Forwarder from transactron.utils.transactron_helpers import make_layout +from transactron.lib import logging class WishboneParameters: @@ -108,6 +109,8 @@ class WishboneMaster(Component): ---------- wb_params: WishboneParameters Parameters for bus generation. + name: str, optional + Name of this bus. Used for logging. Attributes ---------- @@ -125,8 +128,9 @@ class WishboneMaster(Component): wb_master: WishboneInterface - def __init__(self, wb_params: WishboneParameters): + def __init__(self, wb_params: WishboneParameters, name: str = ""): super().__init__({"wb_master": Out(WishboneSignature(wb_params))}) + self.name = name self.wb_params = wb_params self.method_layouts = WishboneMasterMethodLayout(wb_params) @@ -137,6 +141,11 @@ def __init__(self, wb_params: WishboneParameters): # latched input signals self.txn_req = Signal(self.method_layouts.request_layout) + logger_name = "bus.wishbone" + if name != "": + logger_name += f".{name}" + self.log = logging.HardwareLogger(logger_name) + def elaborate(self, platform): m = TModule() @@ -189,7 +198,17 @@ def FSMWBCycStart(request): # noqa: N802 @def_method(m, self.result) def _(): - return result.read(m) + ret = result.read(m) + + self.log.debug( + m, + True, + "response data=0x{:x} err={}", + ret.data, + ret.err, + ) + + return ret @def_method(m, self.request, ready=request_ready & result.write.ready) def _(arg): @@ -197,6 +216,16 @@ def _(arg): # do WBCycStart state in the same clock cycle FSMWBCycStart(arg) + self.log.debug( + m, + True, + "request addr=0x{:x} data=0x{:x} sel=0x{:x} write={}", + arg.addr, + arg.data, + arg.sel, + arg.we, + ) + result.write.schedule_before(self.request) result.read.schedule_before(self.request) diff --git a/pytest.ini b/pytest.ini index 142b00abe..970b444e5 100644 --- a/pytest.ini +++ b/pytest.ini @@ -6,3 +6,4 @@ norecursedirs = '*.egg', '.*', 'build', 'dist', 'venv', '__traces__', '__pycache filterwarnings = ignore:cannot collect test class 'TestbenchIO':pytest.PytestCollectionWarning ignore:No files were found in testpaths:pytest.PytestConfigWarning: +log_cli=true diff --git a/scripts/run_benchmarks.py b/scripts/run_benchmarks.py index 0dc25098a..442cb26ec 100755 --- a/scripts/run_benchmarks.py +++ b/scripts/run_benchmarks.py @@ -74,7 +74,7 @@ def run_benchmarks_with_cocotb(benchmarks: list[str], traces: bool) -> bool: return res.returncode == 0 -def run_benchmarks_with_pysim(benchmarks: list[str], traces: bool, verbose: bool) -> bool: +def run_benchmarks_with_pysim(benchmarks: list[str], traces: bool) -> bool: suite = unittest.TestSuite() def _gen_test(test_name: str): @@ -82,9 +82,7 @@ def test_fn(): traces_file = None if traces: traces_file = "benchmark." + test_name - asyncio.run( - test.regression.benchmark.run_benchmark(PySimulation(verbose, traces_file=traces_file), test_name) - ) + asyncio.run(test.regression.benchmark.run_benchmark(PySimulation(traces_file=traces_file), test_name)) test_fn.__name__ = test_name test_fn.__qualname__ = test_name @@ -94,17 +92,17 @@ def test_fn(): for test_name in benchmarks: suite.addTest(unittest.FunctionTestCase(_gen_test(test_name))) - runner = unittest.TextTestRunner(verbosity=(2 if verbose else 1)) + runner = unittest.TextTestRunner(verbosity=2) result = runner.run(suite) return result.wasSuccessful() -def run_benchmarks(benchmarks: list[str], backend: Literal["pysim", "cocotb"], traces: bool, verbose: bool) -> bool: +def run_benchmarks(benchmarks: list[str], backend: Literal["pysim", "cocotb"], traces: bool) -> bool: if backend == "cocotb": return run_benchmarks_with_cocotb(benchmarks, traces) elif backend == "pysim": - return run_benchmarks_with_pysim(benchmarks, traces, verbose) + return run_benchmarks_with_pysim(benchmarks, traces) return False @@ -144,8 +142,9 @@ def main(): parser = argparse.ArgumentParser() parser.add_argument("-l", "--list", action="store_true", help="List all benchmarks") parser.add_argument("-t", "--trace", action="store_true", help="Dump waveforms") + parser.add_argument("--log-level", default="WARNING", action="store", help="Level of messages to display.") + parser.add_argument("--log-filter", default=".*", action="store", help="Regexp used to filter out logs.") parser.add_argument("-p", "--profile", action="store_true", help="Write execution profiles") - parser.add_argument("-v", "--verbose", action="store_true", help="Verbose output") parser.add_argument("-b", "--backend", default="cocotb", choices=["cocotb", "pysim"], help="Simulation backend") parser.add_argument( "-o", @@ -164,6 +163,9 @@ def main(): print(name) return + os.environ["__TRANSACTRON_LOG_LEVEL"] = args.log_level + os.environ["__TRANSACTRON_LOG_FILTER"] = args.log_filter + if args.benchmark_name: pattern = re.compile(args.benchmark_name) benchmarks = [name for name in benchmarks if pattern.search(name)] @@ -175,7 +177,7 @@ def main(): if args.profile: os.environ["__TRANSACTRON_PROFILE"] = "1" - success = run_benchmarks(benchmarks, args.backend, args.trace, args.verbose) + success = run_benchmarks(benchmarks, args.backend, args.trace) if not success: print("Benchmark execution failed") sys.exit(1) diff --git a/scripts/run_signature.py b/scripts/run_signature.py index b9b9c1701..2e047f9ae 100755 --- a/scripts/run_signature.py +++ b/scripts/run_signature.py @@ -45,32 +45,31 @@ def run_with_cocotb(test_name: str, traces: bool, output: str) -> bool: return os.path.isfile(output) # completed successfully if signature file was created -def run_with_pysim(test_name: str, traces: bool, verbose: bool, output: str) -> bool: +def run_with_pysim(test_name: str, traces: bool, output: str) -> bool: traces_file = None if traces: traces_file = os.path.basename(test_name) try: - asyncio.run( - test.regression.signature.run_test(PySimulation(verbose, traces_file=traces_file), test_name, output) - ) + asyncio.run(test.regression.signature.run_test(PySimulation(traces_file=traces_file), test_name, output)) except RuntimeError as e: print("RuntimeError:", e) return False return True -def run_test(test: str, backend: Literal["pysim", "cocotb"], traces: bool, verbose: bool, output: str) -> bool: +def run_test(test: str, backend: Literal["pysim", "cocotb"], traces: bool, output: str) -> bool: if backend == "cocotb": return run_with_cocotb(test, traces, output) elif backend == "pysim": - return run_with_pysim(test, traces, verbose, output) + return run_with_pysim(test, traces, output) return False def main(): parser = argparse.ArgumentParser() parser.add_argument("-t", "--trace", action="store_true", help="Dump waveforms") - parser.add_argument("-v", "--verbose", action="store_true", help="Verbose output") + parser.add_argument("--log-level", default="WARNING", action="store", help="Level of messages to display.") + parser.add_argument("--log-filter", default=".*", action="store", help="Regexp used to filter out logs.") parser.add_argument("-b", "--backend", default="pysim", choices=["cocotb", "pysim"], help="Simulation backend") parser.add_argument("-o", "--output", default=None, help="Selects output file to write test signature to") parser.add_argument("path") @@ -79,7 +78,10 @@ def main(): output = args.output if args.output else args.path + ".signature" - success = run_test(args.path, args.backend, args.trace, args.verbose, output) + os.environ["__TRANSACTRON_LOG_LEVEL"] = args.log_level + os.environ["__TRANSACTRON_LOG_FILTER"] = args.log_filter + + success = run_test(args.path, args.backend, args.trace, output) if not success: print(f"{args.path}: Program execution failed") diff --git a/test/conftest.py b/test/conftest.py index d0a77f9ec..a291b488d 100644 --- a/test/conftest.py +++ b/test/conftest.py @@ -28,6 +28,7 @@ def pytest_addoption(parser: pytest.Parser): type=int, help="Number of tests to start. If less than number of all selected tests, then starts only subset of them.", ) + group.addoption("--coreblocks-log-filter", default=".*", action="store", help="Regexp used to filter out logs.") def generate_unittestname(item: pytest.Item) -> str: @@ -104,3 +105,9 @@ def pytest_runtest_setup(item: pytest.Item): if item.config.getoption("--coreblocks-profile", False): # type: ignore os.environ["__TRANSACTRON_PROFILE"] = "1" + + log_filter = item.config.getoption("--coreblocks-log-filter") + os.environ["__TRANSACTRON_LOG_FILTER"] = ".*" if not isinstance(log_filter, str) else log_filter + + log_level = item.config.getoption("--log-level") + os.environ["__TRANSACTRON_LOG_LEVEL"] = "WARNING" if not isinstance(log_level, str) else log_level diff --git a/test/regression/cocotb.py b/test/regression/cocotb.py index e68c6f9ca..444360d04 100644 --- a/test/regression/cocotb.py +++ b/test/regression/cocotb.py @@ -1,5 +1,6 @@ from decimal import Decimal import inspect +import re import os from typing import Any from collections.abc import Coroutine @@ -87,10 +88,6 @@ async def start(self): sig_s = WishboneSlaveSignals() if sig_m.we: - cocotb.logging.debug( - f"Wishbone bus '{self.name}' write request: " - f"addr=0x{addr:x} data=0x{int(sig_m.dat_w):x} sel={sig_m.sel}" - ) resp = self.model.write( WriteRequest( addr=addr, @@ -100,7 +97,6 @@ async def start(self): ) ) else: - cocotb.logging.debug(f"Wishbone bus '{self.name}' read request: addr=0x{addr:x} sel={sig_m.sel}") resp = self.model.read( ReadRequest( addr=addr, @@ -123,11 +119,6 @@ async def start(self): raise ValueError("Bus doesn't support rty") sig_s.rty = 1 - cocotb.logging.debug( - f"Wishbone bus '{self.name}' response: " - f"ack={sig_s.ack} err={sig_s.err} rty={sig_s.rty} data={int(sig_s.dat_r):x}" - ) - for _ in range(self.delay): await clock_edge_event # type: ignore @@ -148,6 +139,11 @@ def __init__(self, dut): self.gen_info = GenerationInfo.decode(gen_info_path) + self.log_level = os.environ["__TRANSACTRON_LOG_LEVEL"] + self.log_filter = os.environ["__TRANSACTRON_LOG_FILTER"] + + cocotb.logging.getLogger().setLevel(self.log_level) + def get_cocotb_handle(self, path_components: list[str]) -> ModifiableObject: obj = self.dut # Skip the first component, as it is already referenced in "self.dut" @@ -188,14 +184,40 @@ async def profile_handler(self, clock, profile: Profile): await clock_edge_event # type: ignore - async def assert_handler(self, clock): + async def logging_handler(self, clock): clock_edge_event = FallingEdge(clock) + log_level = cocotb.logging.getLogger().level + + logs = [ + (rec, self.get_cocotb_handle(rec.trigger_location)) + for rec in self.gen_info.logs + if rec.level >= log_level and re.search(self.log_filter, rec.logger_name) + ] + while True: - for assert_info in self.gen_info.asserts: - assert_val = self.get_cocotb_handle(assert_info.location) - n, i = assert_info.src_loc - assert assert_val.value, f"Assertion at {n}:{i}" + for rec, trigger_handle in logs: + if not trigger_handle.value: + continue + + values: list[int] = [] + for field in rec.fields_location: + values.append(int(self.get_cocotb_handle(field).value)) + + formatted_msg = rec.format(*values) + + cocotb_log = cocotb.logging.getLogger(rec.logger_name) + + cocotb_log.log( + rec.level, + "%s:%d] %s", + rec.location[0], + rec.location[1], + formatted_msg, + ) + + if rec.level >= cocotb.logging.ERROR: + assert False, f"Assertion failed at {rec.location[0], rec.location[1]}: {formatted_msg}" await clock_edge_event # type: ignore @@ -219,7 +241,7 @@ async def run(self, mem_model: CoreMemoryModel, timeout_cycles: int = 5000) -> S profile.transactions_and_methods = self.gen_info.profile_data.transactions_and_methods cocotb.start_soon(self.profile_handler(self.dut.clk, profile)) - cocotb.start_soon(self.assert_handler(self.dut.clk)) + cocotb.start_soon(self.logging_handler(self.dut.clk)) success = True try: @@ -236,7 +258,7 @@ async def run(self, mem_model: CoreMemoryModel, timeout_cycles: int = 5000) -> S for reg_name, reg_loc in metric_loc.regs.items(): value = int(self.get_cocotb_handle(reg_loc)) result.metric_values[metric_name][reg_name] = value - cocotb.logging.debug(f"Metric {metric_name}/{reg_name}={value}") + cocotb.logging.info(f"Metric {metric_name}/{reg_name}={value}") return result diff --git a/test/regression/cocotb/benchmark_entrypoint.py b/test/regression/cocotb/benchmark_entrypoint.py index fb3fa59c5..d700b3a4e 100644 --- a/test/regression/cocotb/benchmark_entrypoint.py +++ b/test/regression/cocotb/benchmark_entrypoint.py @@ -1,5 +1,4 @@ import sys -import cocotb from pathlib import Path top_dir = Path(__file__).parent.parent.parent.parent @@ -10,7 +9,6 @@ async def _do_benchmark(dut, benchmark_name): - cocotb.logging.getLogger().setLevel(cocotb.logging.INFO) await run_benchmark(CocotbSimulation(dut), benchmark_name) diff --git a/test/regression/cocotb/signature_entrypoint.py b/test/regression/cocotb/signature_entrypoint.py index 1508502fe..4b8a9d212 100644 --- a/test/regression/cocotb/signature_entrypoint.py +++ b/test/regression/cocotb/signature_entrypoint.py @@ -12,8 +12,6 @@ @cocotb.test() async def do_test(dut): - cocotb.logging.getLogger().setLevel(cocotb.logging.INFO) - test_name = os.environ["TESTNAME"] if test_name is None: raise RuntimeError("No ELF file provided") diff --git a/test/regression/cocotb/test_entrypoint.py b/test/regression/cocotb/test_entrypoint.py index 36879b2a6..71b0ed64f 100644 --- a/test/regression/cocotb/test_entrypoint.py +++ b/test/regression/cocotb/test_entrypoint.py @@ -1,5 +1,4 @@ import sys -import cocotb from pathlib import Path top_dir = Path(__file__).parent.parent.parent.parent @@ -14,7 +13,6 @@ async def do_test(dut, test_name): - cocotb.logging.getLogger().setLevel(cocotb.logging.INFO) if test_name == empty_testcase_name: return await run_test(CocotbSimulation(dut), test_name) diff --git a/test/regression/pysim.py b/test/regression/pysim.py index 7eccf3c9d..804687bba 100644 --- a/test/regression/pysim.py +++ b/test/regression/pysim.py @@ -1,5 +1,6 @@ import re import os +import logging from amaranth.sim import Passive, Settle from amaranth.utils import exact_log2 @@ -10,7 +11,14 @@ from .memory import * from .common import SimulationBackend, SimulationExecutionResult -from transactron.testing import PysimSimulator, TestGen, profiler_process, Profile +from transactron.testing import ( + PysimSimulator, + TestGen, + profiler_process, + Profile, + make_logging_process, + parse_logging_level, +) from transactron.utils.dependencies import DependencyContext, DependencyManager from transactron.lib.metrics import HardwareMetricsManager from ..peripherals.test_wishbone import WishboneInterfaceWrapper @@ -22,13 +30,15 @@ class PySimulation(SimulationBackend): - def __init__(self, verbose: bool, traces_file: Optional[str] = None): + def __init__(self, traces_file: Optional[str] = None): self.gp = GenParams(full_core_config) self.running = False self.cycle_cnt = 0 - self.verbose = verbose self.traces_file = traces_file + self.log_level = parse_logging_level(os.environ["__TRANSACTRON_LOG_LEVEL"]) + self.log_filter = os.environ["__TRANSACTRON_LOG_FILTER"] + self.metrics_manager = HardwareMetricsManager() def _wishbone_slave( @@ -49,17 +59,11 @@ def f(): resp_data = 0 - bus_name = "instr" if is_instr_bus else "data" - if (yield wb_ctrl.wb.we): - if self.verbose: - print(f"Wishbone '{bus_name}' bus write request: addr=0x{addr:x} data={dat_w:x} sel={sel:b}") resp = mem_model.write( WriteRequest(addr=addr, data=dat_w, byte_count=word_width_bytes, byte_sel=sel) ) else: - if self.verbose: - print(f"Wishbone '{bus_name}' bus read request: addr=0x{addr:x} sel={sel:b}") resp = mem_model.read( ReadRequest( addr=addr, @@ -70,9 +74,6 @@ def f(): ) resp_data = resp.data - if self.verbose: - print(f"Wishbone '{bus_name}' bus read response: data=0x{resp.data:x}") - ack = err = rty = 0 match resp.status: case ReplyStatus.OK: @@ -102,8 +103,7 @@ def f(): return f def pretty_dump_metrics(self, metric_values: dict[str, dict[str, int]], filter_regexp: str = ".*"): - print() - print("=== Core metrics dump ===") + str = "=== Core metrics dump ===\n" put_space_before = True for metric_name in sorted(metric_values.keys()): @@ -114,21 +114,23 @@ def pretty_dump_metrics(self, metric_values: dict[str, dict[str, int]], filter_r if metric.description != "": if not put_space_before: - print() + str += "\n" - print(f"# {metric.description}") + str += f"# {metric.description}\n" for reg in metric.regs.values(): reg_value = metric_values[metric_name][reg.name] desc = f" # {reg.description} [reg width={reg.width}]" - print(f"{metric_name}/{reg.name} {reg_value}{desc}") + str += f"{metric_name}/{reg.name} {reg_value}{desc}\n" put_space_before = False if metric.description != "": - print() + str += "\n" put_space_before = True + logging.info(str) + async def run(self, mem_model: CoreMemoryModel, timeout_cycles: int = 5000) -> SimulationExecutionResult: with DependencyContext(DependencyManager()): wb_instr_bus = WishboneSignature(self.gp.wb_params).create() @@ -145,6 +147,11 @@ async def run(self, mem_model: CoreMemoryModel, timeout_cycles: int = 5000) -> S sim.add_sync_process(self._wishbone_slave(mem_model, wb_instr_ctrl, is_instr_bus=True)) sim.add_sync_process(self._wishbone_slave(mem_model, wb_data_ctrl, is_instr_bus=False)) + def on_error(): + raise RuntimeError("Simulation finished due to an error") + + sim.add_sync_process(make_logging_process(self.log_level, self.log_filter, on_error)) + profile = None if "__TRANSACTRON_PROFILE" in os.environ: transaction_manager = DependencyContext.get().get_dependency(TransactionManagerKey()) @@ -166,8 +173,7 @@ def on_sim_finish(): sim.add_sync_process(self._waiter(on_finish=on_sim_finish)) success = sim.run() - if self.verbose: - self.pretty_dump_metrics(metric_values) + self.pretty_dump_metrics(metric_values) return SimulationExecutionResult(success, metric_values, profile) diff --git a/test/regression/test_regression.py b/test/regression/test_regression.py index 53d1d2e95..88fc538f1 100644 --- a/test/regression/test_regression.py +++ b/test/regression/test_regression.py @@ -85,7 +85,7 @@ def regression_body_with_pysim(test_name: str, traces: bool): traces_file = None if traces: traces_file = REGRESSION_TESTS_PREFIX + test_name - asyncio.run(run_test(PySimulation(verbose=False, traces_file=traces_file), test_name)) + asyncio.run(run_test(PySimulation(traces_file=traces_file), test_name)) @pytest.fixture(scope="session") diff --git a/test/transactron/testing/test_assertion.py b/test/transactron/testing/test_assertion.py deleted file mode 100644 index 4becf3062..000000000 --- a/test/transactron/testing/test_assertion.py +++ /dev/null @@ -1,32 +0,0 @@ -from amaranth import * - -from transactron.utils import assertion -from transactron.testing import TestCaseWithSimulator - - -class AssertionTest(Elaboratable): - def __init__(self): - self.input = Signal() - self.output = Signal() - - def elaborate(self, platform): - m = Module() - - m.d.comb += self.output.eq(self.input & ~self.input) - - assertion(m, self.input == self.output) - - return m - - -class TestAssertion(TestCaseWithSimulator): - def test_assertion(self): - m = AssertionTest() - - def proc(): - yield - yield m.input.eq(1) - - with self.assertRaises(AssertionError): - with self.run_simulation(m) as sim: - sim.add_sync_process(proc) diff --git a/test/transactron/testing/test_log.py b/test/transactron/testing/test_log.py new file mode 100644 index 000000000..69f537fdd --- /dev/null +++ b/test/transactron/testing/test_log.py @@ -0,0 +1,124 @@ +from amaranth import * + +from transactron import * +from transactron.testing import TestCaseWithSimulator +from transactron.lib import logging + +LOGGER_NAME = "test_logger" + +log = logging.HardwareLogger(LOGGER_NAME) + + +class LogTest(Elaboratable): + def __init__(self): + self.input = Signal(range(100)) + self.counter = Signal(range(200)) + + def elaborate(self, platform): + m = TModule() + + with m.If(self.input == 42): + log.warning(m, True, "Log triggered under Amaranth If value+3=0x{:x}", self.input + 3) + + log.warning(m, self.input[0] == 0, "Input is even! input={}, counter={}", self.input, self.counter) + + m.d.sync += self.counter.eq(self.counter + 1) + + return m + + +class ErrorLogTest(Elaboratable): + def __init__(self): + self.input = Signal() + self.output = Signal() + + def elaborate(self, platform): + m = TModule() + + m.d.comb += self.output.eq(self.input & ~self.input) + + log.error( + m, + self.input != self.output, + "Input is different than output! input=0x{:x} output=0x{:x}", + self.input, + self.output, + ) + + return m + + +class AssertionTest(Elaboratable): + def __init__(self): + self.input = Signal() + self.output = Signal() + + def elaborate(self, platform): + m = TModule() + + m.d.comb += self.output.eq(self.input & ~self.input) + + log.assertion(m, self.input == self.output, "Output differs") + + return m + + +class TestLog(TestCaseWithSimulator): + def test_log(self): + m = LogTest() + + def proc(): + for i in range(50): + yield + yield m.input.eq(i) + + with self.assertLogs(LOGGER_NAME) as logs: + with self.run_simulation(m) as sim: + sim.add_sync_process(proc) + + self.assertIn( + "WARNING:test_logger:test/transactron/testing/test_log.py:21] Log triggered under Amaranth If value+3=0x2d", + logs.output, + ) + for i in range(0, 50, 2): + expected_msg = ( + "WARNING:test_logger:test/transactron/testing/test_log.py:23] " + + f"Input is even! input={i}, counter={i + 2}" + ) + self.assertIn( + expected_msg, + logs.output, + ) + + def test_error_log(self): + m = ErrorLogTest() + + def proc(): + yield + yield m.input.eq(1) + + with self.assertLogs(LOGGER_NAME) as logs: + with self.assertRaises(AssertionError): + with self.run_simulation(m) as sim: + sim.add_sync_process(proc) + + extected_out = ( + "ERROR:test_logger:test/transactron/testing/test_log.py:40] " + + "Input is different than output! input=0x1 output=0x0" + ) + self.assertIn(extected_out, logs.output) + + def test_assertion(self): + m = AssertionTest() + + def proc(): + yield + yield m.input.eq(1) + + with self.assertLogs(LOGGER_NAME) as logs: + with self.assertRaises(AssertionError): + with self.run_simulation(m) as sim: + sim.add_sync_process(proc) + + extected_out = "ERROR:test_logger:test/transactron/testing/test_log.py:61] Output differs" + self.assertIn(extected_out, logs.output) diff --git a/transactron/lib/logging.py b/transactron/lib/logging.py new file mode 100644 index 000000000..7eb06deb1 --- /dev/null +++ b/transactron/lib/logging.py @@ -0,0 +1,229 @@ +import os +import re +import operator +import logging +from functools import reduce +from dataclasses import dataclass, field +from dataclasses_json import dataclass_json +from typing import TypeAlias + +from amaranth import * +from amaranth.tracer import get_src_loc + +from transactron.utils import SrcLoc +from transactron.utils._typing import ModuleLike, ValueLike +from transactron.utils.dependencies import DependencyContext, ListKey + +LogLevel: TypeAlias = int + + +@dataclass_json +@dataclass +class LogRecordInfo: + """Simulator-backend-agnostic information about a log record that can + be serialized and used outside the Amaranth context. + + Attributes + ---------- + logger_name: str + + level: LogLevel + The severity level of the log. + format_str: str + The template of the message. Should follow PEP 3101 standard. + location: SrcLoc + Source location of the log. + """ + + logger_name: str + level: LogLevel + format_str: str + location: SrcLoc + + def format(self, *args) -> str: + """Format the log message with a set of concrete arguments.""" + + return self.format_str.format(*args) + + +@dataclass +class LogRecord(LogRecordInfo): + """A LogRecord instance represents an event being logged. + + Attributes + ---------- + trigger: Signal + Amaranth signal triggering the log. + fields: Signal + Amaranth signals that will be used to format the message. + """ + + trigger: Signal + fields: list[Signal] = field(default_factory=list) + + +@dataclass(frozen=True) +class LogKey(ListKey[LogRecord]): + pass + + +class HardwareLogger: + """A class for creating log messages in the hardware. + + Intuitively, the hardware logger works similarly to a normal software + logger. You can log a message anywhere in the circuit, but due to the + parallel nature of the hardware you must specify a special trigger signal + which will indicate if a message shall be reported in that cycle. + + Hardware logs are evaluated and printed during simulation, so both + the trigger and the format fields are Amaranth values, i.e. + signals or arbitrary Amaranth expressions. + + Instances of the HardwareLogger class represent a logger for a single + submodule of the circuit. Exactly how a "submodule" is defined is up + to the developer. Submodule are identified by a unique string and + the names can be nested. Names are organized into a namespace hierarchy + where levels are separated by periods, much like the Python package + namespace. So in the instance, submodules names might be "frontend" + for the upper level, and "frontend.icache" and "frontend.bpu" for + the sub-levels. There is no arbitrary limit to the depth of nesting. + + Attributes + ---------- + name: str + Name of this logger. + """ + + def __init__(self, name: str): + """ + Parameters + ---------- + name: str + Name of this logger. Hierarchy levels are separated by periods, + e.g. "backend.fu.jumpbranch". + """ + self.name = name + + def log(self, m: ModuleLike, level: LogLevel, trigger: ValueLike, format: str, *args, src_loc_at: int = 0): + """Registers a hardware log record with the given severity. + + Parameters + ---------- + m: ModuleLike + The module for which the log record is added. + trigger: ValueLike + If the value of this Amaranth expression is true, the log will reported. + format: str + The format of the message as defined in PEP 3101. + *args + Amaranth values that will be read during simulation and used to format + the message. + src_loc_at: int, optional + How many stack frames below to look for the source location, used to + identify the failing assertion. + """ + + def local_src_loc(src_loc: SrcLoc): + return (os.path.relpath(src_loc[0]), src_loc[1]) + + src_loc = local_src_loc(get_src_loc(src_loc_at + 1)) + + trigger_signal = Signal() + m.d.comb += trigger_signal.eq(trigger) + + record = LogRecord( + logger_name=self.name, level=level, format_str=format, location=src_loc, trigger=trigger_signal + ) + + for arg in args: + sig = Signal.like(arg) + m.d.top_comb += sig.eq(arg) + record.fields.append(sig) + + dependencies = DependencyContext.get() + dependencies.add_dependency(LogKey(), record) + + def debug(self, m: ModuleLike, trigger: ValueLike, format: str, *args, **kwargs): + """Log a message with severity 'DEBUG'. + + See `HardwareLogger.log` function for more details. + """ + self.log(m, logging.DEBUG, trigger, format, *args, **kwargs) + + def info(self, m: ModuleLike, trigger: ValueLike, format: str, *args, **kwargs): + """Log a message with severity 'INFO'. + + See `HardwareLogger.log` function for more details. + """ + self.log(m, logging.INFO, trigger, format, *args, **kwargs) + + def warning(self, m: ModuleLike, trigger: ValueLike, format: str, *args, **kwargs): + """Log a message with severity 'WARNING'. + + See `HardwareLogger.log` function for more details. + """ + self.log(m, logging.WARNING, trigger, format, *args, **kwargs) + + def error(self, m: ModuleLike, trigger: ValueLike, format: str, *args, **kwargs): + """Log a message with severity 'ERROR'. + + This severity level has special semantics. If a log with this serverity + level is triggered, the simulation will be terminated. + + See `HardwareLogger.log` function for more details. + """ + self.log(m, logging.ERROR, trigger, format, *args, **kwargs) + + def assertion(self, m: ModuleLike, value: Value, format: str = "", *args, src_loc_at: int = 0, **kwargs): + """Define an assertion. + + This function might help find some hardware bugs which might otherwise be + hard to detect. If `value` is false, it will terminate the simulation or + it can also be used to turn on a warning LED on a board. + + Internally, this is a convenience wrapper over log.error. + + See `HardwareLogger.log` function for more details. + """ + self.error(m, ~value, format, *args, **kwargs, src_loc_at=src_loc_at + 1) + + +def get_log_records(level: LogLevel, namespace_regexp: str = ".*") -> list[LogRecord]: + """Get log records in for the given severity level and in the + specified namespace. + + This function returns all log records with the severity bigger or equal + to the specified level and belonging to the specified namespace. + + Parameters + ---------- + level: LogLevel + The minimum severity level. + namespace: str, optional + The regexp of the namespace. If not specified, logs from all namespaces + will be processed. + """ + + dependencies = DependencyContext.get() + all_logs = dependencies.get_dependency(LogKey()) + return [rec for rec in all_logs if rec.level >= level and re.search(namespace_regexp, rec.logger_name)] + + +def get_trigger_bit(level: LogLevel, namespace_regexp: str = ".*") -> Value: + """Get a trigger bit for logs of the given severity level and + in the specified namespace. + + The signal returned by this function is high whenever the trigger signal + of any of the records with the severity bigger or equal to the specified + level is high. + + Parameters + ---------- + level: LogLevel + The minimum severity level. + namespace: str, optional + The regexp of the namespace. If not specified, logs from all namespaces + will be processed. + """ + + return reduce(operator.or_, [rec.trigger for rec in get_log_records(level, namespace_regexp)], C(0)) diff --git a/transactron/testing/__init__.py b/transactron/testing/__init__.py index 2ed73b0ff..bc5d38fa2 100644 --- a/transactron/testing/__init__.py +++ b/transactron/testing/__init__.py @@ -3,5 +3,5 @@ from .sugar import * # noqa: F401 from .testbenchio import * # noqa: F401 from .profiler import * # noqa: F401 -from .assertion import * # noqa: F401 +from .logging import * # noqa: F401 from transactron.utils import data_layout # noqa: F401 diff --git a/transactron/testing/assertion.py b/transactron/testing/assertion.py deleted file mode 100644 index 19c5a4149..000000000 --- a/transactron/testing/assertion.py +++ /dev/null @@ -1,20 +0,0 @@ -from collections.abc import Callable -from typing import Any -from amaranth.sim import Passive, Tick -from transactron.utils import assert_bit, assert_bits - - -__all__ = ["make_assert_handler"] - - -def make_assert_handler(my_assert: Callable[[int, str], Any]): - def assert_handler(): - yield Passive() - while True: - yield Tick("sync_neg") - if not (yield assert_bit()): - for v, (n, i) in assert_bits(): - my_assert((yield v), f"Assertion at {n}:{i}") - yield - - return assert_handler diff --git a/transactron/testing/infrastructure.py b/transactron/testing/infrastructure.py index dc4a5404e..a769bba13 100644 --- a/transactron/testing/infrastructure.py +++ b/transactron/testing/infrastructure.py @@ -13,7 +13,7 @@ from .testbenchio import TestbenchIO from .profiler import profiler_process, Profile from .functions import TestGen -from .assertion import make_assert_handler +from .logging import make_logging_process, parse_logging_level from .gtkw_extension import write_vcd_ext from transactron import Method from transactron.lib import AdapterTrans @@ -263,7 +263,12 @@ def run_simulation(self, module: HasElaborate, max_cycles: float = 10e4, add_tra profiler_process(sim.tested_module.manager.get_dependency(TransactionManagerKey()), profile) ) - sim.add_sync_process(make_assert_handler(self.assertTrue)) + def on_error(): + self.assertTrue(False, "Simulation finished due to an error") + + log_level = parse_logging_level(os.environ["__TRANSACTRON_LOG_LEVEL"]) + log_filter = os.environ["__TRANSACTRON_LOG_FILTER"] + sim.add_sync_process(make_logging_process(log_level, log_filter, on_error)) res = sim.run() diff --git a/transactron/testing/logging.py b/transactron/testing/logging.py new file mode 100644 index 000000000..6a2ad0881 --- /dev/null +++ b/transactron/testing/logging.py @@ -0,0 +1,108 @@ +from collections.abc import Callable +from typing import Any +import logging + +from amaranth.sim import Passive, Tick +from transactron.lib import logging as tlog + + +__all__ = ["make_logging_process", "parse_logging_level"] + + +def parse_logging_level(str: str) -> tlog.LogLevel: + """Parse the log level from a string. + + The level can be either a non-negative integer or a string representation + of one of the predefined levels. + + Raises an exception if the level cannot be parsed. + """ + str = str.upper() + names_mapping = logging.getLevelNamesMapping() + if str in names_mapping: + return names_mapping[str] + + # try convert to int + try: + return int(str) + except ValueError: + pass + + raise ValueError("Log level must be either {error, warn, info, debug} or a non-negative integer.") + + +_sim_cycle = 0 + + +class _LogFormatter(logging.Formatter): + """ + Log formatter to provide colors and to inject simulator times into + the log messages. Adapted from https://stackoverflow.com/a/56944256/3638629 + """ + + magenta = "\033[0;35m" + grey = "\033[0;34m" + blue = "\033[0;34m" + yellow = "\033[0;33m" + red = "\033[0;31m" + reset = "\033[0m" + + loglevel2colour = { + logging.DEBUG: grey + "{}" + reset, + logging.INFO: magenta + "{}" + reset, + logging.WARNING: yellow + "{}" + reset, + logging.ERROR: red + "{}" + reset, + } + + def format(self, record: logging.LogRecord): + level_name = self.loglevel2colour[record.levelno].format(record.levelname) + return f"{_sim_cycle} {level_name} {record.name} {record.getMessage()}" + + +def make_logging_process(level: tlog.LogLevel, namespace_regexp: str, on_error: Callable[[], Any]): + combined_trigger = tlog.get_trigger_bit(level, namespace_regexp) + records = tlog.get_log_records(level, namespace_regexp) + + root_logger = logging.getLogger() + ch = logging.StreamHandler() + formatter = _LogFormatter() + ch.setFormatter(formatter) + root_logger.handlers = [ch] + + def handle_logs(): + if not (yield combined_trigger): + return + + for record in records: + if not (yield record.trigger): + continue + + values: list[int] = [] + for field in record.fields: + values.append((yield field)) + + formatted_msg = record.format(*values) + + logger = root_logger.getChild(record.logger_name) + logger.log( + record.level, + "%s:%d] %s", + record.location[0], + record.location[1], + formatted_msg, + ) + + if record.level >= logging.ERROR: + on_error() + + def log_process(): + global _sim_cycle + + yield Passive() + while True: + yield Tick("sync_neg") + yield from handle_logs() + yield + _sim_cycle += 1 + + return log_process diff --git a/transactron/utils/__init__.py b/transactron/utils/__init__.py index 703b2aad9..ebf845b7d 100644 --- a/transactron/utils/__init__.py +++ b/transactron/utils/__init__.py @@ -4,7 +4,6 @@ from .assign import * # noqa: F401 from .amaranth_ext import * # noqa: F401 from .transactron_helpers import * # noqa: F401 -from .assertion import * # noqa: F401 from .dependencies import * # noqa: F401 from .depcache import * # noqa: F401 from .idgen import * # noqa: F401 diff --git a/transactron/utils/assertion.py b/transactron/utils/assertion.py deleted file mode 100644 index b79a74fef..000000000 --- a/transactron/utils/assertion.py +++ /dev/null @@ -1,60 +0,0 @@ -from amaranth import * -from amaranth.tracer import get_src_loc -from functools import reduce -import operator -from dataclasses import dataclass -from transactron.utils import SrcLoc -from transactron.utils._typing import ModuleLike, ValueLike -from transactron.utils.dependencies import DependencyContext, ListKey - -__all__ = ["AssertKey", "assertion", "assert_bit", "assert_bits"] - - -@dataclass(frozen=True) -class AssertKey(ListKey[tuple[Signal, SrcLoc]]): - pass - - -def assertion(m: ModuleLike, value: ValueLike, *, src_loc_at: int = 0): - """Define an assertion. - - This function might help find some hardware bugs which might otherwise be - hard to detect. If `value` is false on any assertion, the value returned - from the `assert_bit` function is false. This terminates the simulation, - it can also be used to turn on a warning LED on a board. - - Parameters - ---------- - m: Module - Module in which the assertion is defined. - value : Value - If the value of this Amaranth expression is false, the assertion will - fail. - src_loc_at : int, optional - How many stack frames below to look for the source location, used to - identify the failing assertion. - """ - src_loc = get_src_loc(src_loc_at) - sig = Signal() - m.d.comb += sig.eq(value) - dependencies = DependencyContext.get() - dependencies.add_dependency(AssertKey(), (sig, src_loc)) - - -def assert_bits() -> list[tuple[Signal, SrcLoc]]: - """Gets assertion bits. - - This function returns all the assertion signals created by `assertion`, - together with their source locations. - """ - dependencies = DependencyContext.get() - return dependencies.get_dependency(AssertKey()) - - -def assert_bit() -> Signal: - """Gets assertion bit. - - The signal returned by this function is false if and only if there exists - a false signal among assertion bits created by `assertion`. - """ - return reduce(operator.and_, [a[0] for a in assert_bits()], C(1)) diff --git a/transactron/utils/gen.py b/transactron/utils/gen.py index f87706750..daf462ce7 100644 --- a/transactron/utils/gen.py +++ b/transactron/utils/gen.py @@ -1,5 +1,6 @@ from dataclasses import dataclass, field from dataclasses_json import dataclass_json +from typing import TypeAlias from amaranth import * from amaranth.back import verilog @@ -7,11 +8,10 @@ from transactron.core import TransactionManager, MethodMap, TransactionManagerKey from transactron.lib.metrics import HardwareMetricsManager +from transactron.lib import logging from transactron.utils.dependencies import DependencyContext from transactron.utils.idgen import IdGenerator from transactron.profiler import ProfileData -from transactron.utils._typing import SrcLoc -from transactron.utils.assertion import assert_bits from typing import TYPE_CHECKING @@ -21,11 +21,16 @@ __all__ = [ "MetricLocation", - "AssertLocation", + "GeneratedLog", "GenerationInfo", "generate_verilog", ] +SignalHandle: TypeAlias = list[str] +"""The location of a signal is a list of Verilog identifiers that denote a path +consisting of module names (and the signal name at the end) leading +to the signal wire.""" + @dataclass_json @dataclass @@ -34,13 +39,11 @@ class MetricLocation: Attributes ---------- - regs : dict[str, list[str]] - The location of each register of that metric. The location is a list of - Verilog identifiers that denote a path consiting of modules names - (and the signal name at the end) leading to the register wire. + regs : dict[str, SignalHandle] + The location of each register of that metric. """ - regs: dict[str, list[str]] = field(default_factory=dict) + regs: dict[str, SignalHandle] = field(default_factory=dict) @dataclass_json @@ -79,21 +82,19 @@ class MethodSignalsLocation: @dataclass_json @dataclass -class AssertLocation: - """Information about an assert signal in the generated Verilog code. +class GeneratedLog(logging.LogRecordInfo): + """Information about a log record in the generated Verilog code. Attributes ---------- - location : list[str] - The location of the assert signal. The location is a list of Verilog - identifiers that denote a path consisting of module names (and the - signal name at the end) leading to the signal wire. - src_loc : SrcLoc - Source location of the assertion. + trigger_location : SignalHandle + The location of the trigger signal. + fields_location : list[SignalHandle] + Locations of the log fields. """ - location: list[str] - src_loc: SrcLoc + trigger_location: SignalHandle + fields_location: list[SignalHandle] @dataclass_json @@ -106,16 +107,15 @@ class GenerationInfo: metrics_location : dict[str, MetricInfo] Mapping from a metric name to an object storing Verilog locations of its registers. - asserts : list[AssertLocation] - Locations and metadata for assertion signals. + logs : list[GeneratedLog] + Locations and metadata for all log records. """ metrics_location: dict[str, MetricLocation] transaction_signals_location: dict[int, TransactionSignalsLocation] method_signals_location: dict[int, MethodSignalsLocation] profile_data: ProfileData - metrics_location: dict[str, MetricLocation] - asserts: list[AssertLocation] + logs: list[GeneratedLog] def encode(self, file_name: str): """ @@ -158,7 +158,7 @@ def escape_verilog_identifier(identifier: str) -> str: return identifier -def get_signal_location(signal: Signal, name_map: "SignalDict") -> list[str]: +def get_signal_location(signal: Signal, name_map: "SignalDict") -> SignalHandle: raw_location = name_map[signal] return raw_location @@ -204,13 +204,24 @@ def collect_transaction_method_signals( return (transaction_signals_location, method_signals_location) -def collect_asserts(name_map: "SignalDict") -> list[AssertLocation]: - asserts: list[AssertLocation] = [] - - for v, src_loc in assert_bits(): - asserts.append(AssertLocation(get_signal_location(v, name_map), src_loc)) +def collect_logs(name_map: "SignalDict") -> list[GeneratedLog]: + logs: list[GeneratedLog] = [] + + # Get all records. + for record in logging.get_log_records(0): + trigger_loc = get_signal_location(record.trigger, name_map) + fields_loc = [get_signal_location(field, name_map) for field in record.fields] + log = GeneratedLog( + logger_name=record.logger_name, + level=record.level, + format_str=record.format_str, + location=record.location, + trigger_location=trigger_loc, + fields_location=fields_loc, + ) + logs.append(log) - return asserts + return logs def generate_verilog( @@ -229,7 +240,7 @@ def generate_verilog( transaction_signals_location=transaction_signals, method_signals_location=method_signals, profile_data=profile_data, - asserts=collect_asserts(name_map), + logs=collect_logs(name_map), ) return verilog_text, gen_info