diff --git a/b_asic/architecture.py b/b_asic/architecture.py new file mode 100644 index 0000000000000000000000000000000000000000..5622ef8c94f2247992f99923521f46b0780bc251 --- /dev/null +++ b/b_asic/architecture.py @@ -0,0 +1,134 @@ +""" +B-ASIC architecture classes. +""" +from typing import Set, cast + +from b_asic.process import MemoryVariable, OperatorProcess, PlainMemoryVariable +from b_asic.resources import ProcessCollection + + +class ProcessingElement: + """ + Create a processing element for a ProcessCollection with OperatorProcesses. + + Parameters + ---------- + process_collection : :class:`~b_asic.resources.ProcessCollection` + """ + + def __init__(self, process_collection: ProcessCollection): + if not len(ProcessCollection): + raise ValueError( + "Do not create ProcessingElement with empty ProcessCollection" + ) + if not all( + isinstance(operator, OperatorProcess) + for operator in process_collection.collection + ): + raise TypeError( + "Can only have OperatorProcesses in ProcessCollection when creating" + " ProcessingElement" + ) + ops = [ + cast(operand, OperatorProcess).operation + for operand in process_collection.collection + ] + op_type = type(ops[0]) + if not all(isinstance(op, op_type) for op in ops): + raise TypeError("Different Operation types in ProcessCollection") + self._collection = process_collection + self._operation_type = op_type + self._type_name = op_type.type_name() + + def write_code(self, path: str, entity_name: str) -> None: + """ + Write VHDL code for processing element. + + Parameters + ---------- + path : str + Directory to write code in. + entity_name : str + """ + raise NotImplementedError + + +class Memory: + """ + Create a memory from a ProcessCollection with memory variables. + + Parameters + ---------- + process_collection : :class:`~b_asic.resources.ProcessCollection` + The ProcessCollection to create a Memory for. + memory_type : {'RAM', 'register'} + The type of memory. + """ + + def __init__(self, process_collection: ProcessCollection, memory_type: str = "RAM"): + if not len(ProcessCollection): + raise ValueError("Do not create Memory with empty ProcessCollection") + if not all( + isinstance(operator, (MemoryVariable, PlainMemoryVariable)) + for operator in process_collection.collection + ): + raise TypeError( + "Can only have MemoryVariable or PlainMemoryVariable in" + " ProcessCollection when creating Memory" + ) + self._collection = process_collection + self._memory_type = memory_type + + def write_code(self, path: str, entity_name: str) -> None: + """ + Write VHDL code for memory. + + Parameters + ---------- + path : str + Directory to write code in. + entity_name : str + + Returns + ------- + + + """ + raise NotImplementedError + + +class Architecture: + """ + Class representing an architecture. + + Parameters + ---------- + processing_elements : set of :class:`~b_asic.architecture.ProcessingElement` + The processing elements in the architecture. + memories : set of :class:`~b_asic.architecture.Memory` + The memories in the architecture. + name : str, default: "arch" + Name for the top-level architecture. Used for the entity and as prefix for all + building blocks. + """ + + def __init__( + self, + processing_elements: Set[ProcessingElement], + memories: Set[Memory], + name: str = "arch", + ): + self._processing_elements = processing_elements + self._memories = memories + self._name = name + + def write_code(self, path: str) -> None: + """ + Write HDL of architecture. + + Parameters + ---------- + path : str + Directory to write code in. + """ + raise NotImplementedError diff --git a/b_asic/process.py b/b_asic/process.py index 93343d410614d96aa73221fe642f1446f65ecded..161a6af9eb06ecb50d6ef76de93ea9c1bba2afab 100644 --- a/b_asic/process.py +++ b/b_asic/process.py @@ -96,6 +96,14 @@ class OperatorProcess(Process): ) self._operation = operation + @property + def operation(self) -> Operation: + """The Operation that the OperatorProcess corresponds to.""" + return self._operation + + def __repr__(self) -> str: + return f"OperatorProcess({self.start_time}, {self.operation}, {self.name!r})" + class MemoryVariable(Process): """ diff --git a/b_asic/resources.py b/b_asic/resources.py index 82b5dc922d13994ed018fcd09973d18a719a3797..4111c7206d2eeeccf03bd941935171fba53620b4 100644 --- a/b_asic/resources.py +++ b/b_asic/resources.py @@ -1001,9 +1001,7 @@ class ProcessCollection: for collection in assignment: for mv in collection: if mv not in self: - raise ValueError( - f'{mv.__repr__()} is not part of {self.__repr__()}.' - ) + raise ValueError(f'{mv!r} is not part of {self!r}.') # Make sure that concurrent reads/writes do not surpass the port setting for mv in self: @@ -1141,7 +1139,7 @@ class ProcessCollection: process for process in self._collection if isinstance(process, OperatorProcess) - and process._operation.type_name() == type_name + and process.operation.type_name() == type_name }, self._schedule_time, self._cyclic, diff --git a/docs_sphinx/api/architecture.rst b/docs_sphinx/api/architecture.rst new file mode 100644 index 0000000000000000000000000000000000000000..4e1330b4ef502ca547ef03165e183d381f9044e1 --- /dev/null +++ b/docs_sphinx/api/architecture.rst @@ -0,0 +1,7 @@ +*********************** +``b_asic.architecture`` +*********************** + +.. automodule:: b_asic.architecture + :members: + :undoc-members: diff --git a/docs_sphinx/api/index.rst b/docs_sphinx/api/index.rst index bb4c23eb7f194358c4403546ed49bde56be91229..484dfd3b104578a508dc5a9c383ee08306e1d025 100644 --- a/docs_sphinx/api/index.rst +++ b/docs_sphinx/api/index.rst @@ -5,6 +5,7 @@ API .. toctree:: :maxdepth: 1 + architecture.rst core_operations.rst graph_component.rst operation.rst diff --git a/test/fixtures/schedule.py b/test/fixtures/schedule.py index c5a78609776642dd6db4cc3b425897a07160b770..adabad2e71f8f3eae5b0bc1bdd8cca30078fcc1d 100644 --- a/test/fixtures/schedule.py +++ b/test/fixtures/schedule.py @@ -11,3 +11,16 @@ def secondorder_iir_schedule(precedence_sfg_delays): schedule = Schedule(precedence_sfg_delays, scheduling_algorithm="ASAP") return schedule + + +@pytest.fixture +def secondorder_iir_schedule_with_execution_times(precedence_sfg_delays): + precedence_sfg_delays.set_latency_of_type(Addition.type_name(), 4) + precedence_sfg_delays.set_latency_of_type(ConstantMultiplication.type_name(), 3) + precedence_sfg_delays.set_execution_time_of_type(Addition.type_name(), 2) + precedence_sfg_delays.set_execution_time_of_type( + ConstantMultiplication.type_name(), 1 + ) + + schedule = Schedule(precedence_sfg_delays, scheduling_algorithm="ASAP") + return schedule diff --git a/test/test_process.py b/test/test_process.py index ea53284889ee43bdbcfd75fc313a3d7a50c1297e..f02fe882a621dc968d97b179392b7d3847149fa0 100644 --- a/test/test_process.py +++ b/test/test_process.py @@ -1,5 +1,7 @@ import re +import pytest + from b_asic.process import PlainMemoryVariable @@ -32,3 +34,8 @@ def test_MemoryVariables(secondorder_iir_schedule): assert pattern.match(repr(mem_var)) assert mem_var.execution_time == 4 assert mem_var.start_time == 3 + + +def test_OperatorProcess_error(secondorder_iir_schedule): + with pytest.raises(ValueError, match="does not have an execution time specified"): + _ = secondorder_iir_schedule.get_operations() diff --git a/test/test_resources.py b/test/test_resources.py index 9575d85f6aebd390f48e18644ab28e214081429c..86616a2340a6e3dc04b5b542c3543c272e60ad30 100644 --- a/test/test_resources.py +++ b/test/test_resources.py @@ -4,6 +4,7 @@ import re import matplotlib.pyplot as plt import pytest +from b_asic.core_operations import ConstantMultiplication from b_asic.process import PlainMemoryVariable from b_asic.research.interleaver import ( generate_matrix_transposer, @@ -124,3 +125,12 @@ class TestProcessCollectionPlainMemoryVariable: def test_len_process_collection(self, simple_collection: ProcessCollection): assert len(simple_collection) == 7 + + def test_get_by_type_name(self, secondorder_iir_schedule_with_execution_times): + pc = secondorder_iir_schedule_with_execution_times.get_operations() + pc_cmul = pc.get_by_type_name(ConstantMultiplication.type_name()) + assert len(pc_cmul) == 7 + assert all( + isinstance(operand.operation, ConstantMultiplication) + for operand in pc_cmul.collection + ) diff --git a/test/test_schedule.py b/test/test_schedule.py index 70ab9e9ff5eb32bb36eb77147e12a4b7d79bd26b..3a7059b3bc33f8c987243550c46d1529e7b3cff0 100644 --- a/test/test_schedule.py +++ b/test/test_schedule.py @@ -6,6 +6,7 @@ import re import pytest from b_asic.core_operations import Addition, Butterfly, ConstantMultiplication +from b_asic.process import OperatorProcess from b_asic.schedule import Schedule from b_asic.signal_flow_graph import SFG from b_asic.special_operations import Delay, Input, Output @@ -562,3 +563,8 @@ class TestErrors: NotImplementedError, match="No algorithm with name: foo defined." ): Schedule(sfg_simple_filter, scheduling_algorithm="foo") + + def test_get_operations(self, secondorder_iir_schedule_with_execution_times): + pc = secondorder_iir_schedule_with_execution_times.get_operations() + assert len(pc) == 13 + assert all(isinstance(operand, OperatorProcess) for operand in pc.collection)