From 752b9861109bbad66ae88a4ee24f7c7180b4aa3b Mon Sep 17 00:00:00 2001 From: lekcyjna123 <34948061+lekcyjna123@users.noreply.github.com> Date: Mon, 13 Nov 2023 00:20:35 +0100 Subject: [PATCH 1/2] Move coreblocks utils to transactron (#503) Co-authored-by: Lekcyjna <309016@uwr.edu.pl> Co-authored-by: Marek Materzok --- coreblocks/core.py | 2 +- coreblocks/frontend/fetch.py | 2 +- coreblocks/frontend/icache.py | 4 ++-- coreblocks/frontend/rvc.py | 2 +- coreblocks/fu/alu.py | 4 ++-- coreblocks/fu/div_unit.py | 4 ++-- coreblocks/fu/exception.py | 2 +- coreblocks/fu/jumpbranch.py | 2 +- coreblocks/fu/mul_unit.py | 2 +- coreblocks/fu/shift_unit.py | 2 +- coreblocks/fu/zbc.py | 2 +- coreblocks/fu/zbs.py | 2 +- coreblocks/lsu/dummyLsu.py | 2 +- coreblocks/params/instr.py | 2 +- coreblocks/params/layouts.py | 2 +- coreblocks/peripherals/wishbone.py | 4 ++-- coreblocks/scheduler/scheduler.py | 2 +- coreblocks/scheduler/wakeup_select.py | 2 +- coreblocks/structs_common/csr.py | 2 +- coreblocks/utils/__init__.py | 3 --- coreblocks/utils/protocols.py | 2 +- scripts/gen_verilog.py | 2 +- scripts/synthesize.py | 2 +- test/common/functions.py | 2 +- test/common/infrastructure.py | 2 +- test/common/sugar.py | 2 +- test/common/testbenchio.py | 2 +- test/frontend/test_fetch.py | 2 +- test/frontend/test_rvc.py | 2 +- test/gtkw_extension.py | 2 +- test/regression/memory.py | 2 +- test/structs_common/test_exception.py | 2 +- test/test_core.py | 2 +- test/transactions/test_assign.py | 4 ++-- test/transactions/test_simultaneous.py | 2 +- test/transactions/test_transaction_lib.py | 4 ++-- test/utils/test_fifo.py | 2 +- test/utils/test_onehotswitch.py | 2 +- test/utils/test_utils.py | 2 +- transactron/_utils.py | 4 ++-- transactron/core.py | 7 +++---- transactron/lib/reqres.py | 2 +- transactron/lib/simultaneous.py | 2 +- transactron/lib/storage.py | 2 +- transactron/lib/transformers.py | 2 +- transactron/tracing.py | 2 +- transactron/utils/__init__.py | 3 +++ {coreblocks => transactron}/utils/_typing.py | 0 {coreblocks => transactron}/utils/debug_signals.py | 0 {coreblocks => transactron}/utils/fifo.py | 2 +- {coreblocks => transactron}/utils/utils.py | 0 51 files changed, 58 insertions(+), 59 deletions(-) create mode 100644 transactron/utils/__init__.py rename {coreblocks => transactron}/utils/_typing.py (100%) rename {coreblocks => transactron}/utils/debug_signals.py (100%) rename {coreblocks => transactron}/utils/fifo.py (99%) rename {coreblocks => transactron}/utils/utils.py (100%) diff --git a/coreblocks/core.py b/coreblocks/core.py index 409ab3238..1cec20059 100644 --- a/coreblocks/core.py +++ b/coreblocks/core.py @@ -20,7 +20,7 @@ from coreblocks.frontend.icache import ICache, SimpleWBCacheRefiller, ICacheBypass from coreblocks.peripherals.wishbone import WishboneMaster, WishboneBus from coreblocks.frontend.fetch import Fetch, UnalignedFetch -from coreblocks.utils.fifo import BasicFifo +from transactron.utils.fifo import BasicFifo __all__ = ["Core"] diff --git a/coreblocks/frontend/fetch.py b/coreblocks/frontend/fetch.py index 3b1090b51..74c3861a1 100644 --- a/coreblocks/frontend/fetch.py +++ b/coreblocks/frontend/fetch.py @@ -1,5 +1,5 @@ from amaranth import * -from coreblocks.utils.fifo import BasicFifo, Semaphore +from transactron.utils.fifo import BasicFifo, Semaphore from coreblocks.frontend.icache import ICacheInterface from coreblocks.frontend.rvc import InstrDecompress, is_instr_compressed from transactron import def_method, Method, Transaction, TModule diff --git a/coreblocks/frontend/icache.py b/coreblocks/frontend/icache.py index d5cbb8d8a..6fd7d38d1 100644 --- a/coreblocks/frontend/icache.py +++ b/coreblocks/frontend/icache.py @@ -8,8 +8,8 @@ from transactron.core import def_method, Priority, TModule from transactron import Method, Transaction from coreblocks.params import ICacheLayouts, ICacheParameters -from coreblocks.utils import assign, OneHotSwitchDynamic -from coreblocks.utils._typing import HasElaborate +from transactron.utils import assign, OneHotSwitchDynamic +from transactron.utils._typing import HasElaborate from transactron.lib import * from coreblocks.peripherals.wishbone import WishboneMaster diff --git a/coreblocks/frontend/rvc.py b/coreblocks/frontend/rvc.py index 17be75d62..ad5156ebc 100644 --- a/coreblocks/frontend/rvc.py +++ b/coreblocks/frontend/rvc.py @@ -2,7 +2,7 @@ from transactron import TModule from coreblocks.params import * -from coreblocks.utils import ValueLike +from transactron.utils import ValueLike # An instruction or an instruction with the valid signal diff --git a/coreblocks/fu/alu.py b/coreblocks/fu/alu.py index 76de2188f..f9a2bad21 100644 --- a/coreblocks/fu/alu.py +++ b/coreblocks/fu/alu.py @@ -5,14 +5,14 @@ from transactron.lib import FIFO from coreblocks.params import OpType, Funct3, Funct7, GenParams, FuncUnitLayouts, FunctionalComponentParams -from coreblocks.utils import HasElaborate, OneHotSwitch +from transactron.utils import HasElaborate, OneHotSwitch from coreblocks.fu.fu_decoder import DecoderManager from enum import IntFlag, auto from coreblocks.utils.protocols import FuncUnit -from coreblocks.utils.utils import popcount, count_leading_zeros +from transactron.utils.utils import popcount, count_leading_zeros __all__ = ["AluFuncUnit", "ALUComponent"] diff --git a/coreblocks/fu/div_unit.py b/coreblocks/fu/div_unit.py index c874d439f..a39390af8 100644 --- a/coreblocks/fu/div_unit.py +++ b/coreblocks/fu/div_unit.py @@ -12,8 +12,8 @@ from coreblocks.fu.fu_decoder import DecoderManager -from coreblocks.utils import OneHotSwitch -from coreblocks.utils.fifo import BasicFifo +from transactron.utils import OneHotSwitch +from transactron.utils.fifo import BasicFifo from coreblocks.utils.protocols import FuncUnit from coreblocks.fu.division.long_division import LongDivider diff --git a/coreblocks/fu/exception.py b/coreblocks/fu/exception.py index bca24f08b..4ef733e8c 100644 --- a/coreblocks/fu/exception.py +++ b/coreblocks/fu/exception.py @@ -7,7 +7,7 @@ from transactron.lib import FIFO from coreblocks.params import OpType, GenParams, FuncUnitLayouts, FunctionalComponentParams -from coreblocks.utils import OneHotSwitch +from transactron.utils import OneHotSwitch from coreblocks.params.keys import ExceptionReportKey from coreblocks.fu.fu_decoder import DecoderManager diff --git a/coreblocks/fu/jumpbranch.py b/coreblocks/fu/jumpbranch.py index dcd5f0936..b42320a23 100644 --- a/coreblocks/fu/jumpbranch.py +++ b/coreblocks/fu/jumpbranch.py @@ -9,7 +9,7 @@ from transactron.lib import * from coreblocks.params import * -from coreblocks.utils import OneHotSwitch +from transactron.utils import OneHotSwitch from coreblocks.utils.protocols import FuncUnit from coreblocks.fu.fu_decoder import DecoderManager diff --git a/coreblocks/fu/mul_unit.py b/coreblocks/fu/mul_unit.py index 3d4888cfa..47f9e9f6e 100644 --- a/coreblocks/fu/mul_unit.py +++ b/coreblocks/fu/mul_unit.py @@ -18,7 +18,7 @@ __all__ = ["MulUnit", "MulFn", "MulComponent", "MulType"] -from coreblocks.utils import OneHotSwitch +from transactron.utils import OneHotSwitch from coreblocks.utils.protocols import FuncUnit diff --git a/coreblocks/fu/shift_unit.py b/coreblocks/fu/shift_unit.py index 14a8ae6e7..7e73ae2f4 100644 --- a/coreblocks/fu/shift_unit.py +++ b/coreblocks/fu/shift_unit.py @@ -5,7 +5,7 @@ from transactron.lib import FIFO from coreblocks.params import OpType, Funct3, Funct7, GenParams, FuncUnitLayouts, FunctionalComponentParams -from coreblocks.utils import OneHotSwitch +from transactron.utils import OneHotSwitch from coreblocks.fu.fu_decoder import DecoderManager from enum import IntFlag, auto diff --git a/coreblocks/fu/zbc.py b/coreblocks/fu/zbc.py index bba84ddb3..8834d40fe 100644 --- a/coreblocks/fu/zbc.py +++ b/coreblocks/fu/zbc.py @@ -14,7 +14,7 @@ ) from transactron import Method, def_method, TModule from transactron.lib import FIFO -from coreblocks.utils import OneHotSwitch +from transactron.utils import OneHotSwitch from coreblocks.utils.protocols import FuncUnit diff --git a/coreblocks/fu/zbs.py b/coreblocks/fu/zbs.py index fa0b9e86e..d135c5e75 100644 --- a/coreblocks/fu/zbs.py +++ b/coreblocks/fu/zbs.py @@ -5,7 +5,7 @@ from coreblocks.params import Funct3, GenParams, FuncUnitLayouts, OpType, Funct7, FunctionalComponentParams from transactron import Method, TModule, def_method from transactron.lib import FIFO -from coreblocks.utils import OneHotSwitch +from transactron.utils import OneHotSwitch from coreblocks.utils.protocols import FuncUnit from coreblocks.fu.fu_decoder import DecoderManager diff --git a/coreblocks/lsu/dummyLsu.py b/coreblocks/lsu/dummyLsu.py index 497ef3355..cb506aafc 100644 --- a/coreblocks/lsu/dummyLsu.py +++ b/coreblocks/lsu/dummyLsu.py @@ -3,7 +3,7 @@ from transactron import Method, def_method, Transaction, TModule from coreblocks.params import * from coreblocks.peripherals.wishbone import WishboneMaster -from coreblocks.utils import assign, ModuleLike +from transactron.utils import assign, ModuleLike from coreblocks.utils.protocols import FuncBlock diff --git a/coreblocks/params/instr.py b/coreblocks/params/instr.py index 5e77763fa..ae207b1dd 100644 --- a/coreblocks/params/instr.py +++ b/coreblocks/params/instr.py @@ -3,7 +3,7 @@ from amaranth.hdl.ast import ValueCastable from amaranth import * -from coreblocks.utils import ValueLike +from transactron.utils import ValueLike from coreblocks.params.isa import * diff --git a/coreblocks/params/layouts.py b/coreblocks/params/layouts.py index 9386d11e0..2a0f1a3e0 100644 --- a/coreblocks/params/layouts.py +++ b/coreblocks/params/layouts.py @@ -1,6 +1,6 @@ from coreblocks.params import GenParams, OpType, Funct7, Funct3 from coreblocks.params.isa import ExceptionCause -from coreblocks.utils.utils import layout_subset +from transactron.utils.utils import layout_subset __all__ = [ "SchedulerLayouts", diff --git a/coreblocks/peripherals/wishbone.py b/coreblocks/peripherals/wishbone.py index c6035431d..01f80338e 100644 --- a/coreblocks/peripherals/wishbone.py +++ b/coreblocks/peripherals/wishbone.py @@ -7,8 +7,8 @@ from transactron import Method, def_method, TModule from transactron.lib import AdapterTrans -from coreblocks.utils.utils import OneHotSwitchDynamic, assign -from coreblocks.utils.fifo import BasicFifo +from transactron.utils.utils import OneHotSwitchDynamic, assign +from transactron.utils.fifo import BasicFifo class WishboneParameters: diff --git a/coreblocks/scheduler/scheduler.py b/coreblocks/scheduler/scheduler.py index eab680fe2..3fe9ea293 100644 --- a/coreblocks/scheduler/scheduler.py +++ b/coreblocks/scheduler/scheduler.py @@ -5,7 +5,7 @@ from transactron import Method, Transaction, TModule from transactron.lib import FIFO, Forwarder from coreblocks.params import SchedulerLayouts, GenParams, OpType -from coreblocks.utils import assign, AssignType +from transactron.utils import assign, AssignType from coreblocks.utils.protocols import FuncBlock diff --git a/coreblocks/scheduler/wakeup_select.py b/coreblocks/scheduler/wakeup_select.py index 7480a3936..fbe6d40a4 100644 --- a/coreblocks/scheduler/wakeup_select.py +++ b/coreblocks/scheduler/wakeup_select.py @@ -1,7 +1,7 @@ from amaranth import * from coreblocks.params import GenParams, FuncUnitLayouts -from coreblocks.utils import assign, AssignType +from transactron.utils import assign, AssignType from transactron.core import * __all__ = ["WakeupSelect"] diff --git a/coreblocks/structs_common/csr.py b/coreblocks/structs_common/csr.py index 0f8eba862..3dc4763d6 100644 --- a/coreblocks/structs_common/csr.py +++ b/coreblocks/structs_common/csr.py @@ -3,7 +3,7 @@ from dataclasses import dataclass from transactron import Method, def_method, Transaction, TModule -from coreblocks.utils import assign, bits_from_int +from transactron.utils import assign, bits_from_int from coreblocks.params.genparams import GenParams from coreblocks.params.dependencies import DependencyManager, ListKey from coreblocks.params.fu_params import BlockComponentParams diff --git a/coreblocks/utils/__init__.py b/coreblocks/utils/__init__.py index a1cd9d2e0..e69de29bb 100644 --- a/coreblocks/utils/__init__.py +++ b/coreblocks/utils/__init__.py @@ -1,3 +0,0 @@ -from .utils import * # noqa: F401 -from ._typing import * # noqa: F401 -from .debug_signals import * # noqa: F401 diff --git a/coreblocks/utils/protocols.py b/coreblocks/utils/protocols.py index bc7885274..70f542bc2 100644 --- a/coreblocks/utils/protocols.py +++ b/coreblocks/utils/protocols.py @@ -1,6 +1,6 @@ from typing import Protocol from transactron import Method -from ._typing import HasElaborate +from transactron.utils._typing import HasElaborate __all__ = ["FuncUnit", "FuncBlock", "Unifier"] diff --git a/scripts/gen_verilog.py b/scripts/gen_verilog.py index 4f388a825..42bf94e4e 100755 --- a/scripts/gen_verilog.py +++ b/scripts/gen_verilog.py @@ -16,7 +16,7 @@ from coreblocks.peripherals.wishbone import WishboneBus from coreblocks.core import Core from transactron import TransactionModule -from coreblocks.utils.utils import flatten_signals +from transactron.utils.utils import flatten_signals from coreblocks.params.configurations import * diff --git a/scripts/synthesize.py b/scripts/synthesize.py index 3763b921f..572e4b77a 100755 --- a/scripts/synthesize.py +++ b/scripts/synthesize.py @@ -13,7 +13,7 @@ sys.path.insert(0, parent) -from coreblocks.utils.utils import ModuleConnector +from transactron.utils.utils import ModuleConnector from coreblocks.params.genparams import GenParams from coreblocks.params.fu_params import FunctionalComponentParams from coreblocks.core import Core diff --git a/test/common/functions.py b/test/common/functions.py index c4ffc814a..42b43cdf1 100644 --- a/test/common/functions.py +++ b/test/common/functions.py @@ -2,7 +2,7 @@ from amaranth.hdl.ast import Statement from amaranth.sim.core import Command from typing import TypeVar, Any, Generator, TypeAlias -from coreblocks.utils._typing import RecordValueDict, RecordIntDict +from transactron.utils._typing import RecordValueDict, RecordIntDict T = TypeVar("T") TestGen: TypeAlias = Generator[Command | Value | Statement | None, Any, T] diff --git a/test/common/infrastructure.py b/test/common/infrastructure.py index fe0b337d4..e1141a0eb 100644 --- a/test/common/infrastructure.py +++ b/test/common/infrastructure.py @@ -11,7 +11,7 @@ from transactron import Method from transactron.lib import AdapterTrans from transactron.core import TransactionModule -from coreblocks.utils import ModuleConnector, HasElaborate, auto_debug_signals, HasDebugSignals +from transactron.utils import ModuleConnector, HasElaborate, auto_debug_signals, HasDebugSignals T = TypeVar("T") _T_nested_collection = T | list["_T_nested_collection[T]"] | dict[str, "_T_nested_collection[T]"] diff --git a/test/common/sugar.py b/test/common/sugar.py index beb4acf3a..e0ac3473c 100644 --- a/test/common/sugar.py +++ b/test/common/sugar.py @@ -1,7 +1,7 @@ import functools from typing import Callable, Any, Optional from .testbenchio import TestbenchIO, TestGen -from coreblocks.utils._typing import RecordIntDict +from transactron.utils._typing import RecordIntDict def def_method_mock( diff --git a/test/common/testbenchio.py b/test/common/testbenchio.py index 9a2c956d8..0ff3e9d74 100644 --- a/test/common/testbenchio.py +++ b/test/common/testbenchio.py @@ -4,7 +4,7 @@ from transactron.lib import AdapterBase from transactron.core import ValueLike, SignalBundle from transactron._utils import mock_def_helper -from coreblocks.utils._typing import RecordIntDictRet, RecordValueDict, RecordIntDict +from transactron.utils._typing import RecordIntDictRet, RecordValueDict, RecordIntDict from .functions import set_inputs, get_outputs, TestGen diff --git a/test/frontend/test_fetch.py b/test/frontend/test_fetch.py index 9df857ea5..3817c0312 100644 --- a/test/frontend/test_fetch.py +++ b/test/frontend/test_fetch.py @@ -11,7 +11,7 @@ from coreblocks.frontend.icache import ICacheInterface from coreblocks.params import * from coreblocks.params.configurations import test_core_config -from coreblocks.utils import ModuleConnector +from transactron.utils import ModuleConnector from ..common import TestCaseWithSimulator, TestbenchIO, def_method_mock diff --git a/test/frontend/test_rvc.py b/test/frontend/test_rvc.py index 188339cd9..e8d9e4d5b 100644 --- a/test/frontend/test_rvc.py +++ b/test/frontend/test_rvc.py @@ -6,7 +6,7 @@ from coreblocks.frontend.rvc import InstrDecompress from coreblocks.params import * from coreblocks.params.configurations import test_core_config -from coreblocks.utils import ValueLike +from transactron.utils import ValueLike from ..common import TestCaseWithSimulator diff --git a/test/gtkw_extension.py b/test/gtkw_extension.py index 4d8582f93..d548ad6e7 100644 --- a/test/gtkw_extension.py +++ b/test/gtkw_extension.py @@ -2,7 +2,7 @@ from contextlib import contextmanager from amaranth.sim.pysim import _VCDWriter from amaranth import * -from coreblocks.utils.utils import flatten_signals +from transactron.utils.utils import flatten_signals class _VCDWriterExt(_VCDWriter): diff --git a/test/regression/memory.py b/test/regression/memory.py index f1640f71a..38abacd63 100644 --- a/test/regression/memory.py +++ b/test/regression/memory.py @@ -6,7 +6,7 @@ from elftools.elf.constants import P_FLAGS from elftools.elf.elffile import ELFFile, Segment from coreblocks.params.configurations import CoreConfiguration -from coreblocks.utils.utils import align_to_power_of_two, align_down_to_power_of_two +from transactron.utils.utils import align_to_power_of_two, align_down_to_power_of_two all = [ "ReplyStatus", diff --git a/test/structs_common/test_exception.py b/test/structs_common/test_exception.py index f8bde5c56..12238c8f0 100644 --- a/test/structs_common/test_exception.py +++ b/test/structs_common/test_exception.py @@ -6,7 +6,7 @@ from coreblocks.params.isa import ExceptionCause from coreblocks.params.configurations import test_core_config from transactron.lib import Adapter -from coreblocks.utils.utils import ModuleConnector +from transactron.utils.utils import ModuleConnector from ..common import * diff --git a/test/test_core.py b/test/test_core.py index aa03c64e3..a44d92c12 100644 --- a/test/test_core.py +++ b/test/test_core.py @@ -1,7 +1,7 @@ from amaranth import Elaboratable, Module from transactron.lib import AdapterTrans -from coreblocks.utils import align_to_power_of_two +from transactron.utils import align_to_power_of_two from .common import TestCaseWithSimulator, TestbenchIO diff --git a/test/transactions/test_assign.py b/test/transactions/test_assign.py index 6f0569dbc..7f472b57f 100644 --- a/test/transactions/test_assign.py +++ b/test/transactions/test_assign.py @@ -3,8 +3,8 @@ from amaranth.lib import data from amaranth.hdl.ast import ArrayProxy, Slice -from coreblocks.utils._typing import LayoutLike -from coreblocks.utils.utils import AssignArg, AssignType, AssignFields, assign +from transactron.utils._typing import LayoutLike +from transactron.utils.utils import AssignArg, AssignType, AssignFields, assign from unittest import TestCase from parameterized import parameterized_class, parameterized diff --git a/test/transactions/test_simultaneous.py b/test/transactions/test_simultaneous.py index ac3009ac7..0f33d2021 100644 --- a/test/transactions/test_simultaneous.py +++ b/test/transactions/test_simultaneous.py @@ -3,7 +3,7 @@ from amaranth import * from amaranth.sim import * -from coreblocks.utils.utils import ModuleConnector +from transactron.utils.utils import ModuleConnector from ..common import SimpleTestCircuit, TestCaseWithSimulator, TestbenchIO, def_method_mock diff --git a/test/transactions/test_transaction_lib.py b/test/transactions/test_transaction_lib.py index a23ee077e..d43540860 100644 --- a/test/transactions/test_transaction_lib.py +++ b/test/transactions/test_transaction_lib.py @@ -13,8 +13,8 @@ from transactron.core import RecordDict from transactron.lib import * from coreblocks.utils import * -from coreblocks.utils._typing import LayoutLike, ModuleLike -from coreblocks.utils import ModuleConnector +from transactron.utils._typing import LayoutLike, ModuleLike +from transactron.utils import ModuleConnector from ..common import ( SimpleTestCircuit, TestCaseWithSimulator, diff --git a/test/utils/test_fifo.py b/test/utils/test_fifo.py index 81c764099..2f5608141 100644 --- a/test/utils/test_fifo.py +++ b/test/utils/test_fifo.py @@ -1,6 +1,6 @@ from amaranth import * -from coreblocks.utils.fifo import BasicFifo +from transactron.utils.fifo import BasicFifo from transactron.lib import AdapterTrans from test.common import TestCaseWithSimulator, TestbenchIO, data_layout diff --git a/test/utils/test_onehotswitch.py b/test/utils/test_onehotswitch.py index d6c70180f..12d8373b9 100644 --- a/test/utils/test_onehotswitch.py +++ b/test/utils/test_onehotswitch.py @@ -1,7 +1,7 @@ from amaranth import * from amaranth.sim import * -from coreblocks.utils import OneHotSwitch +from transactron.utils import OneHotSwitch from test.common import TestCaseWithSimulator diff --git a/test/utils/test_utils.py b/test/utils/test_utils.py index 6a948f490..a26567003 100644 --- a/test/utils/test_utils.py +++ b/test/utils/test_utils.py @@ -3,7 +3,7 @@ from amaranth import * from test.common import * -from coreblocks.utils import ( +from transactron.utils import ( align_to_power_of_two, align_down_to_power_of_two, popcount, diff --git a/transactron/_utils.py b/transactron/_utils.py index bcd60d8db..1ad803e1f 100644 --- a/transactron/_utils.py +++ b/transactron/_utils.py @@ -4,8 +4,8 @@ from typing import Any, Concatenate, Optional, TypeAlias, TypeGuard, TypeVar from collections.abc import Callable, Iterable, Mapping from amaranth import * -from coreblocks.utils._typing import LayoutLike, ShapeLike -from coreblocks.utils import OneHotSwitchDynamic +from transactron.utils._typing import LayoutLike, ShapeLike +from transactron.utils import OneHotSwitchDynamic __all__ = [ "Scheduler", diff --git a/transactron/core.py b/transactron/core.py index 035b3b588..363a59f06 100644 --- a/transactron/core.py +++ b/transactron/core.py @@ -21,11 +21,10 @@ from itertools import count, chain, filterfalse, product from amaranth.hdl.dsl import FSM, _ModuleBuilderDomain -from coreblocks.utils import AssignType, assign, ModuleConnector -from coreblocks.utils.utils import OneHotSwitchDynamic +from transactron.utils import AssignType, assign, ModuleConnector, silence_mustuse +from transactron.utils.utils import OneHotSwitchDynamic from ._utils import * -from coreblocks.utils import silence_mustuse -from coreblocks.utils._typing import ValueLike, SignalBundle, HasElaborate, SwitchKey, ModuleLike +from transactron.utils._typing import ValueLike, SignalBundle, HasElaborate, SwitchKey, ModuleLike from .graph import Owned, OwnershipGraph, Direction __all__ = [ diff --git a/transactron/lib/reqres.py b/transactron/lib/reqres.py index ace563c1b..3587edc3e 100644 --- a/transactron/lib/reqres.py +++ b/transactron/lib/reqres.py @@ -1,7 +1,7 @@ from amaranth import * from ..core import * from .connectors import Forwarder, FIFO -from coreblocks.utils.fifo import BasicFifo +from transactron.utils.fifo import BasicFifo from amaranth.utils import * __all__ = [ diff --git a/transactron/lib/simultaneous.py b/transactron/lib/simultaneous.py index 093bfb78d..48814cde8 100644 --- a/transactron/lib/simultaneous.py +++ b/transactron/lib/simultaneous.py @@ -3,7 +3,7 @@ from ..core import TransactionBase from contextlib import contextmanager from typing import Optional -from coreblocks.utils import ValueLike +from transactron.utils import ValueLike __all__ = [ "condition", diff --git a/transactron/lib/storage.py b/transactron/lib/storage.py index 4723e3713..9d95f6a83 100644 --- a/transactron/lib/storage.py +++ b/transactron/lib/storage.py @@ -2,7 +2,7 @@ from amaranth.utils import * from ..core import * from typing import Optional -from coreblocks.utils import assign, AssignType +from transactron.utils import assign, AssignType from .reqres import ArgumentsToResultsZipper __all__ = ["MemoryBank"] diff --git a/transactron/lib/transformers.py b/transactron/lib/transformers.py index e4b7aa0c0..b3d1e2470 100644 --- a/transactron/lib/transformers.py +++ b/transactron/lib/transformers.py @@ -3,7 +3,7 @@ from ..core import RecordDict from typing import Optional from collections.abc import Callable -from coreblocks.utils import ValueLike, assign, AssignType +from transactron.utils import ValueLike, assign, AssignType from .connectors import Forwarder, ManyToOneConnectTrans, ConnectTrans __all__ = [ diff --git a/transactron/tracing.py b/transactron/tracing.py index 2a944ce6d..036044aed 100644 --- a/transactron/tracing.py +++ b/transactron/tracing.py @@ -7,7 +7,7 @@ from amaranth.hdl.ir import Elaboratable, Fragment, Instance from amaranth.hdl.xfrm import FragmentTransformer from amaranth.hdl import dsl, ir, mem, xfrm -from coreblocks.utils import HasElaborate +from transactron.utils import HasElaborate from . import core diff --git a/transactron/utils/__init__.py b/transactron/utils/__init__.py new file mode 100644 index 000000000..a1cd9d2e0 --- /dev/null +++ b/transactron/utils/__init__.py @@ -0,0 +1,3 @@ +from .utils import * # noqa: F401 +from ._typing import * # noqa: F401 +from .debug_signals import * # noqa: F401 diff --git a/coreblocks/utils/_typing.py b/transactron/utils/_typing.py similarity index 100% rename from coreblocks/utils/_typing.py rename to transactron/utils/_typing.py diff --git a/coreblocks/utils/debug_signals.py b/transactron/utils/debug_signals.py similarity index 100% rename from coreblocks/utils/debug_signals.py rename to transactron/utils/debug_signals.py diff --git a/coreblocks/utils/fifo.py b/transactron/utils/fifo.py similarity index 99% rename from coreblocks/utils/fifo.py rename to transactron/utils/fifo.py index 2f95adf19..94e136e13 100644 --- a/coreblocks/utils/fifo.py +++ b/transactron/utils/fifo.py @@ -1,7 +1,7 @@ from amaranth import * from transactron import Method, def_method, Priority, TModule from transactron._utils import MethodLayout -from coreblocks.utils._typing import ValueLike +from transactron.utils._typing import ValueLike class BasicFifo(Elaboratable): diff --git a/coreblocks/utils/utils.py b/transactron/utils/utils.py similarity index 100% rename from coreblocks/utils/utils.py rename to transactron/utils/utils.py From 78f66cc976cebfb88f907c92e7da2a4af83a9cea Mon Sep 17 00:00:00 2001 From: Marek Materzok Date: Mon, 13 Nov 2023 14:31:17 +0100 Subject: [PATCH 2/2] Faster Wishbone (#505) --- coreblocks/frontend/icache.py | 21 ++++- coreblocks/peripherals/wishbone.py | 55 +++++++----- test/frontend/test_icache.py | 1 + test/peripherals/test_wishbone.py | 137 +++++++++++++++++++++-------- transactron/core.py | 11 ++- 5 files changed, 156 insertions(+), 69 deletions(-) diff --git a/coreblocks/frontend/icache.py b/coreblocks/frontend/icache.py index 6fd7d38d1..576ea1220 100644 --- a/coreblocks/frontend/icache.py +++ b/coreblocks/frontend/icache.py @@ -382,10 +382,15 @@ def elaborate(self, platform): refill_active = Signal() word_counter = Signal(range(self.params.words_in_block)) - with Transaction().body(m, request=refill_active): + m.submodules.address_fwd = address_fwd = Forwarder( + [("word_counter", word_counter.shape()), ("refill_address", refill_address.shape())] + ) + + with Transaction().body(m): + address = address_fwd.read(m) self.wb_master.request( m, - addr=Cat(word_counter, refill_address), + addr=Cat(address["word_counter"], address["refill_address"]), data=0, we=0, sel=Repl(1, self.wb_master.wb_params.data_width // self.wb_master.wb_params.granularity), @@ -393,19 +398,27 @@ def elaborate(self, platform): @def_method(m, self.start_refill, ready=~refill_active) def _(addr) -> None: - m.d.sync += refill_address.eq(addr[self.params.offset_bits :]) + address = addr[self.params.offset_bits :] + m.d.sync += refill_address.eq(address) m.d.sync += refill_active.eq(1) m.d.sync += word_counter.eq(0) + address_fwd.write(m, word_counter=0, refill_address=address) + @def_method(m, self.accept_refill, ready=refill_active) def _(): fetched = self.wb_master.result(m) last = (word_counter == (self.params.words_in_block - 1)) | fetched.err - m.d.sync += word_counter.eq(word_counter + 1) + next_word_counter = Signal.like(word_counter) + m.d.top_comb += next_word_counter.eq(word_counter + 1) + + m.d.sync += word_counter.eq(next_word_counter) with m.If(last): m.d.sync += refill_active.eq(0) + with m.Else(): + address_fwd.write(m, word_counter=next_word_counter, refill_address=refill_address) return { "addr": Cat(Repl(0, log2_int(self.params.word_width_bytes)), word_counter, refill_address), diff --git a/coreblocks/peripherals/wishbone.py b/coreblocks/peripherals/wishbone.py index 01f80338e..ae88e74ec 100644 --- a/coreblocks/peripherals/wishbone.py +++ b/coreblocks/peripherals/wishbone.py @@ -6,9 +6,11 @@ import operator from transactron import Method, def_method, TModule +from transactron.core import Transaction from transactron.lib import AdapterTrans from transactron.utils.utils import OneHotSwitchDynamic, assign from transactron.utils.fifo import BasicFifo +from transactron.lib.connectors import Forwarder class WishboneParameters: @@ -109,8 +111,6 @@ def __init__(self, wb_params: WishboneParameters): self.request = Method(i=self.requestLayout) self.result = Method(o=self.resultLayout) - self.ready = Signal() - self.res_ready = Signal() self.result_data = Record(self.resultLayout) # latched input signals @@ -132,6 +132,10 @@ def generate_layouts(self, wb_params: WishboneParameters): def elaborate(self, platform): m = TModule() + m.submodules.result = result = Forwarder(self.resultLayout) + + request_ready = Signal() + def FSMWBCycStart(request): # noqa: N802 # internal FSM function that starts Wishbone cycle m.d.sync += self.wbMaster.cyc.eq(1) @@ -140,12 +144,6 @@ def FSMWBCycStart(request): # noqa: N802 m.d.sync += self.wbMaster.dat_w.eq(Mux(request.we, request.data, 0)) m.d.sync += self.wbMaster.we.eq(request.we) m.d.sync += self.wbMaster.sel.eq(request.sel) - m.next = "WBWaitACK" - - @def_method(m, self.result, ready=self.res_ready) - def _(): - m.d.sync += self.res_ready.eq(0) - return self.result_data with m.FSM("Reset"): with m.State("Reset"): @@ -153,17 +151,12 @@ def _(): m.next = "Idle" with m.State("Idle"): # default values for important signals - m.d.sync += self.ready.eq(1) m.d.sync += self.wbMaster.rst.eq(0) m.d.sync += self.wbMaster.stb.eq(0) m.d.sync += self.wbMaster.cyc.eq(0) - - @def_method(m, self.request, ready=(self.ready & ~self.res_ready)) - def _(arg): - m.d.sync += self.ready.eq(0) - m.d.sync += assign(self.txn_req, arg) - # do WBCycStart state in the same clock cycle - FSMWBCycStart(arg) + m.d.comb += request_ready.eq(1) + with m.If(self.request.run): + m.next = "WBWaitACK" with m.State("WBCycStart"): FSMWBCycStart(self.txn_req) @@ -171,18 +164,34 @@ def _(arg): with m.State("WBWaitACK"): with m.If(self.wbMaster.ack | self.wbMaster.err): - m.d.sync += self.wbMaster.cyc.eq(0) - m.d.sync += self.wbMaster.stb.eq(0) - m.d.sync += self.ready.eq(1) - m.d.sync += self.res_ready.eq(1) - m.d.sync += self.result_data.data.eq(Mux(self.txn_req.we, 0, self.wbMaster.dat_r)) - m.d.sync += self.result_data.err.eq(self.wbMaster.err) - m.next = "Idle" + m.d.comb += request_ready.eq(result.read.run) + with Transaction().body(m): + # will be always ready, as we checked that in Idle + result.write(m, data=Mux(self.txn_req.we, 0, self.wbMaster.dat_r), err=self.wbMaster.err) + with m.If(self.request.run): + m.next = "WBWaitACK" + with m.Else(): + m.d.sync += self.wbMaster.cyc.eq(0) + m.d.sync += self.wbMaster.stb.eq(0) + m.next = "Idle" with m.If(self.wbMaster.rty): m.d.sync += self.wbMaster.cyc.eq(1) m.d.sync += self.wbMaster.stb.eq(0) m.next = "WBCycStart" + @def_method(m, self.result) + def _(): + return result.read(m) + + @def_method(m, self.request, ready=request_ready & result.write.ready) + def _(arg): + m.d.sync += assign(self.txn_req, arg) + # do WBCycStart state in the same clock cycle + FSMWBCycStart(arg) + + result.write.schedule_before(self.request) + result.read.schedule_before(self.request) + return m diff --git a/test/frontend/test_icache.py b/test/frontend/test_icache.py index c0ac4f800..35fa0a6d8 100644 --- a/test/frontend/test_icache.py +++ b/test/frontend/test_icache.py @@ -92,6 +92,7 @@ def wishbone_slave(self): # Wishbone is addressing words, so we need to shift it a bit to get the real address. addr = (yield self.test_module.wb_ctrl.wb.adr) << log2_int(self.cp.word_width_bytes) + yield while random.random() < 0.5: yield diff --git a/test/peripherals/test_wishbone.py b/test/peripherals/test_wishbone.py index b4a7d78f0..9db5b9430 100644 --- a/test/peripherals/test_wishbone.py +++ b/test/peripherals/test_wishbone.py @@ -36,8 +36,8 @@ def slave_verify(self, exp_addr, exp_data, exp_we, exp_sel=0): assert (yield self.wb.stb) and (yield self.wb.cyc) assert (yield self.wb.adr) == exp_addr - assert (yield self.wb.we == exp_we) - assert (yield self.wb.sel == exp_sel) + assert (yield self.wb.we) == exp_we + assert (yield self.wb.sel) == exp_sel if exp_we: assert (yield self.wb.dat_w) == exp_data @@ -53,6 +53,10 @@ def slave_respond(self, data, ack=1, err=0, rty=0): yield self.wb.err.eq(0) yield self.wb.rty.eq(0) + def wait_ack(self): + while not ((yield self.wb.stb) and (yield self.wb.cyc) and (yield self.wb.ack)): + yield + class TestWishboneMaster(TestCaseWithSimulator): class WishboneMasterTestModule(Elaboratable): @@ -70,43 +74,70 @@ def test_manual(self): twbm = TestWishboneMaster.WishboneMasterTestModule() def process(): - wbm = twbm.wbm - wwb = WishboneInterfaceWrapper(wbm.wbMaster) - # read request yield from twbm.requestAdapter.call(addr=2, data=0, we=0, sel=1) + + # read request after delay yield - assert not (yield wbm.request.ready) - yield from wwb.slave_verify(2, 0, 0, 1) - yield from wwb.slave_respond(8) - resp = yield from twbm.resultAdapter.call() - assert (resp["data"]) == 8 + yield + yield from twbm.requestAdapter.call(addr=1, data=0, we=0, sel=1) # write request yield from twbm.requestAdapter.call(addr=3, data=5, we=1, sel=0) - yield - yield from wwb.slave_verify(3, 5, 1, 0) - yield from wwb.slave_respond(0) - yield from twbm.resultAdapter.call() # RTY and ERR responese yield from twbm.requestAdapter.call(addr=2, data=0, we=0, sel=0) - yield + resp = yield from twbm.requestAdapter.call_try() + self.assertIsNone(resp) # verify cycle restart + + def result_process(): + resp = yield from twbm.resultAdapter.call() + self.assertEqual(resp["data"], 8) + self.assertFalse(resp["err"]) + + resp = yield from twbm.resultAdapter.call() + self.assertEqual(resp["data"], 3) + self.assertFalse(resp["err"]) + + resp = yield from twbm.resultAdapter.call() + self.assertFalse(resp["err"]) + + resp = yield from twbm.resultAdapter.call() + self.assertEqual(resp["data"], 1) + self.assertTrue(resp["err"]) + + def slave(): + wwb = WishboneInterfaceWrapper(twbm.wbm.wbMaster) + + yield from wwb.slave_wait() + yield from wwb.slave_verify(2, 0, 0, 1) + yield from wwb.slave_respond(8) + yield Settle() + yield from wwb.slave_wait() + yield from wwb.slave_verify(1, 0, 0, 1) + yield from wwb.slave_respond(3) + yield Settle() + + yield # consecutive request + yield from wwb.slave_verify(3, 5, 1, 0) + yield from wwb.slave_respond(0) + yield + + yield # consecutive request yield from wwb.slave_verify(2, 0, 0, 0) yield from wwb.slave_respond(1, ack=0, err=0, rty=1) - yield + yield Settle() assert not (yield wwb.wb.stb) - assert not (yield wbm.result.ready) # verify cycle restart + yield from wwb.slave_wait() yield from wwb.slave_verify(2, 0, 0, 0) yield from wwb.slave_respond(1, ack=1, err=1, rty=0) - resp = yield from twbm.resultAdapter.call() - assert resp["data"] == 1 - assert resp["err"] with self.run_simulation(twbm) as sim: sim.add_sync_process(process) + sim.add_sync_process(result_process) + sim.add_sync_process(slave) class TestWishboneMuxer(TestCaseWithSimulator): @@ -295,7 +326,7 @@ def elaborate(self, platform): class TestWishboneMemorySlave(TestCaseWithSimulator): def setUp(self): - self.memsize = 430 # test some weird depth + self.memsize = 43 # test some weird depth self.iters = 300 self.addr_width = (self.memsize - 1).bit_length() # nearest log2 >= log2(memsize) @@ -307,27 +338,55 @@ def setUp(self): random.seed(42) def test_randomized(self): - def mem_op_process(): - mem_state = [0] * self.memsize + req_queue = deque() + wr_queue = deque() + + mem_state = [0] * self.memsize + def request_process(): for _ in range(self.iters): - addr = random.randint(0, self.memsize - 1) - data = random.randint(0, 2**self.wb_params.data_width - 1) - write = random.randint(0, 1) - sel = random.randint(0, 2**self.sel_width - 1) - if write: + req = { + "addr": random.randint(0, self.memsize - 1), + "data": random.randint(0, 2**self.wb_params.data_width - 1), + "we": random.randint(0, 1), + "sel": random.randint(0, 2**self.sel_width - 1), + } + req_queue.appendleft(req) + wr_queue.appendleft(req) + + while random.random() < 0.2: + yield + yield from self.m.request.call(req) + + def result_process(): + for _ in range(self.iters): + while random.random() < 0.2: + yield + res = yield from self.m.result.call() + req = req_queue.pop() + + if not req["we"]: + self.assertEqual(res["data"], mem_state[req["addr"]]) + + def write_process(): + wwb = WishboneInterfaceWrapper(self.m.mem_master.wbMaster) + for _ in range(self.iters): + yield from wwb.wait_ack() + req = wr_queue.pop() + + if req["we"]: for i in range(self.sel_width): - if sel & (1 << i): + if req["sel"] & (1 << i): granularity_mask = (2**self.wb_params.granularity - 1) << (i * self.wb_params.granularity) - mem_state[addr] &= ~granularity_mask - mem_state[addr] |= data & granularity_mask + mem_state[req["addr"]] &= ~granularity_mask + mem_state[req["addr"]] |= req["data"] & granularity_mask - yield from self.m.request.call(addr=addr, data=data, we=write, sel=sel) - res = yield from self.m.result.call() - if write: - self.assertEqual((yield self.m.mem_slave.mem[addr]), mem_state[addr]) - else: - self.assertEqual(res["data"], mem_state[addr]) + yield + + if req["we"]: + self.assertEqual((yield self.m.mem_slave.mem[req["addr"]]), mem_state[req["addr"]]) - with self.run_simulation(self.m, max_cycles=1500) as sim: - sim.add_sync_process(mem_op_process) + with self.run_simulation(self.m, max_cycles=3000) as sim: + sim.add_sync_process(request_process) + sim.add_sync_process(result_process) + sim.add_sync_process(write_process) diff --git a/transactron/core.py b/transactron/core.py index 363a59f06..a4b3f3b52 100644 --- a/transactron/core.py +++ b/transactron/core.py @@ -64,6 +64,7 @@ class RelationBase(TypedDict): end: TransactionOrMethod priority: Priority conflict: bool + silence_warning: bool class Relation(RelationBase): @@ -271,7 +272,7 @@ def add_edge(begin: Transaction, end: Transaction, priority: Priority, conflict: start = relation["start"] end = relation["end"] if not relation["conflict"]: # relation added with schedule_before - if end.def_order < start.def_order: + if end.def_order < start.def_order and not relation["silence_warning"]: raise RuntimeError(f"{start.name!r} scheduled before {end.name!r}, but defined afterwards") for trans_start in method_map.transactions_for(start): @@ -714,7 +715,9 @@ def add_conflict(self, end: TransactionOrMethod, priority: Priority = Priority.U Is one of conflicting `Transaction`\\s or `Method`\\s prioritized? Defaults to undefined priority relation. """ - self.relations.append(RelationBase(end=end, priority=priority, conflict=True)) + self.relations.append( + RelationBase(end=end, priority=priority, conflict=True, silence_warning=self.owner != end.owner) + ) def schedule_before(self, end: TransactionOrMethod) -> None: """Adds a priority relation. @@ -728,7 +731,9 @@ def schedule_before(self, end: TransactionOrMethod) -> None: end: Transaction or Method The other `Transaction` or `Method` """ - self.relations.append(RelationBase(end=end, priority=Priority.LEFT, conflict=False)) + self.relations.append( + RelationBase(end=end, priority=Priority.LEFT, conflict=False, silence_warning=self.owner != end.owner) + ) def use_method(self, method: "Method", arg: ValueLike, enable: ValueLike): if method in self.method_uses: