Skip to content
Snippets Groups Projects
port.py 6.03 KiB
Newer Older
  • Learn to ignore specific revisions
  • """@package docstring
    
    B-ASIC Port Module.
    TODO: More info.
    """
    
    from abc import ABC, abstractmethod
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
    from copy import copy
    from typing import NewType, Optional, List, Iterable, TYPE_CHECKING
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
    if TYPE_CHECKING:
        from b_asic.operation import Operation
    
    class Port(ABC):
    
    Jacob Wahlman's avatar
    Jacob Wahlman committed
        """
    
        @property
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
        def operation(self) -> "Operation":
    
            raise NotImplementedError
    
        @property
        @abstractmethod
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
        def index(self) -> int:
            """Return the index of the port."""
    
    Jacob Wahlman's avatar
    Jacob Wahlman committed
    
        @property
        @abstractmethod
        def signal_count(self) -> int:
    
            """Return the number of connected signals."""
            raise NotImplementedError
    
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
        @property
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
        def signals(self) -> Iterable[Signal]:
            """Return all connected signals."""
    
    Jacob Wahlman's avatar
    Jacob Wahlman committed
            raise NotImplementedError
    
        @abstractmethod
    
        def add_signal(self, signal: Signal) -> None:
    
            """Connect this port to the entered signal. If the entered signal isn't connected to
    
            this port then connect the entered signal to the port aswell.
            """
    
    Jacob Wahlman's avatar
    Jacob Wahlman committed
            raise NotImplementedError
    
    
        def remove_signal(self, signal: Signal) -> None:
            """Remove the signal that was entered from the Ports signals.
            If the entered signal still is connected to this port then disconnect the
            entered signal from the port aswell.
    
            Keyword arguments:
            - signal: Signal to remove.
            """
            raise NotImplementedError
    
        @abstractmethod
        def clear(self) -> None:
            """Removes all connected signals from the Port."""
    
    Jacob Wahlman's avatar
    Jacob Wahlman committed
            raise NotImplementedError
    
    class AbstractPort(Port):
        """Abstract port class.
    
        Handles functionality for port id and saves the connection to the parent operation.
        """
    
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
        _operation: "Operation"
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
        def __init__(self, operation: "Operation", index: int):
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
            self._index = index
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
        def operation(self) -> "Operation":
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
        def index(self) -> int:
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
    class SignalSourceProvider(ABC):
        """Signal source provider interface.
        TODO: More info.
        """
    
        @property
        @abstractmethod
        def source(self) -> "OutputPort":
            """Get the main source port provided by this object."""
            raise NotImplementedError
    
    
    
    Jacob Wahlman's avatar
    Jacob Wahlman committed
        """Input port.
        TODO: More info.
        """
    
        _value_length: Optional[int]
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
        def __init__(self, operation: "Operation", index: int):
            super().__init__(operation, index)
    
            self._value_length = None
    
    Jacob Wahlman's avatar
    Jacob Wahlman committed
        @property
        def signal_count(self) -> int:
    
            return 0 if self._source_signal is None else 1
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
        @property
        def signals(self) -> Iterable[Signal]:
            return [] if self._source_signal is None else [self._source_signal]
    
        def add_signal(self, signal: Signal) -> None:
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
            assert self._source_signal is None, "Input port may have only one signal added."
            assert signal is not self._source_signal, "Attempted to add already connected signal."
            self._source_signal = signal
            signal.set_destination(self)
    
    
        def remove_signal(self, signal: Signal) -> None:
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
            assert signal is self._source_signal, "Attempted to remove already removed signal."
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
            signal.remove_destination()
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
            if self._source_signal is not None:
                self.remove_signal(self._source_signal)
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
        @property
        def connected_source(self) -> Optional["OutputPort"]:
            """Get the output port that is currently connected to this input port,
            or None if it is unconnected.
            """
            return None if self._source_signal is None else self._source_signal.source
    
        def connect(self, src: SignalSourceProvider, name: Name = "") -> Signal:
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
            """Connect the provided signal source to this input port by creating a new signal.
            Returns the new signal.
            """
            assert self._source_signal is None, "Attempted to connect already connected input port."
    
            # self._source_signal is set by the signal constructor.
            return Signal(source=src.source, destination=self, name=name)
    
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
        @property
        def value_length(self) -> Optional[int]:
            """Get the number of bits that this port should truncate received values to."""
            return self._value_length
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
        @value_length.setter
        def value_length(self, bits: Optional[int]) -> None:
            """Set the number of bits that this port should truncate received values to."""
    
            assert bits is None or (isinstance(
                bits, int) and bits >= 0), "Value length must be non-negative."
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
            self._value_length = bits
    
    
    class OutputPort(AbstractPort, SignalSourceProvider):
    
    Jacob Wahlman's avatar
    Jacob Wahlman committed
        """Output port.
        TODO: More info.
        """
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
        def __init__(self, operation: "Operation", index: int):
            super().__init__(operation, index)
    
    Jacob Wahlman's avatar
    Jacob Wahlman committed
        @property
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
        @property
        def signals(self) -> Iterable[Signal]:
            return self._destination_signals
    
        def add_signal(self, signal: Signal) -> None:
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
            assert signal not in self._destination_signals, "Attempted to add already connected signal."
    
            self._destination_signals.append(signal)
    
    
        def remove_signal(self, signal: Signal) -> None:
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
            assert signal in self._destination_signals, "Attempted to remove already removed signal."
            self._destination_signals.remove(signal)
            signal.remove_source()
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
            for signal in copy(self._destination_signals):
    
    Ivar Härnqvist's avatar
    Ivar Härnqvist committed
        @property
        def source(self) -> "OutputPort":