Skip to content
Snippets Groups Projects
port.py 7.5 KiB
Newer Older
  • Learn to ignore specific revisions
  • """
    B-ASIC Port Module.
    
    
    Contains classes for managing the ports of operations.
    
    """
    
    from abc import ABC, abstractmethod
    
    from typing import TYPE_CHECKING, List, Optional, Sequence
    
    Oscar Gustafsson's avatar
    Oscar Gustafsson committed
    from b_asic.signal import Signal
    
    
    if TYPE_CHECKING:
        from b_asic.operation import Operation
    
    
    
    class Port(ABC):
    
        """
        Port interface.
    
        Ports serve as connection points for connecting signals between operations.
        They also store information about the latency of the corresponding
        calculations of the operation.
    
        Aside from connected signals, each port also provides a reference to the
        parent operation that "owns" it as well as the operation's port index at
        which the port resides.
    
        """
    
        @property
        @abstractmethod
    
            """Return the connected operation."""
            raise NotImplementedError
    
        @property
        @abstractmethod
    
        def index(self) -> int:
            """Return the index of the port."""
    
            raise NotImplementedError
    
        @property
        @abstractmethod
    
    Oscar Gustafsson's avatar
    Oscar Gustafsson committed
        def latency_offset(self) -> Optional[int]:
    
            """Get the latency_offset of the port."""
    
            raise NotImplementedError
    
    
        @abstractmethod
    
        def latency_offset(self, latency_offset: int) -> None:
    
            """Set the latency_offset of the port to the integer specified value.
            """
    
            raise NotImplementedError
    
        @property
        @abstractmethod
        def signal_count(self) -> int:
            """Return the number of connected signals."""
            raise NotImplementedError
    
    
        @abstractmethod
    
        def signals(self) -> Sequence[Signal]:
    
            raise NotImplementedError
    
        @abstractmethod
        def add_signal(self, signal: Signal) -> None:
    
            Connect this port to the entered signal. If the entered signal is not
            connected to this port then connect the entered signal to the port as
            well.
    
            raise NotImplementedError
    
        @abstractmethod
        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
    
            Parameters
            ==========
    
            signal : Signal
                Signal to remove.
    
            """
            raise NotImplementedError
    
        @abstractmethod
        def clear(self) -> None:
            """Removes all connected signals from the Port."""
            raise NotImplementedError
    
    
    class AbstractPort(Port):
    
        """
        Generic abstract port base class.
    
        Concrete ports should normally derive from this to get the default
        behavior.
    
        _index: int
    
    Oscar Gustafsson's avatar
    Oscar Gustafsson committed
        def __init__(
            self,
            operation: "Operation",
            index: int,
            latency_offset: Optional[int] = None,
        ):
    
            """Construct a port of the given operation at the given port index."""
    
            self._operation = operation
    
            self._index = index
            self._latency_offset = latency_offset
    
    
        @property
    
            return self._operation
    
        @property
    
            return self._index
    
    
    Oscar Gustafsson's avatar
    Oscar Gustafsson committed
        def latency_offset(self) -> Optional[int]:
    
            return self._latency_offset
    
        @latency_offset.setter
        def latency_offset(self, latency_offset: int):
            self._latency_offset = latency_offset
    
    
    class SignalSourceProvider(ABC):
    
        """
        Signal source provider interface.
    
    
        Signal source providers give access to a single output port that can be
        used to connect signals from.
        """
    
        @property
        @abstractmethod
        def source(self) -> "OutputPort":
            """Get the main source port provided by this object."""
            raise NotImplementedError
    
    
    
    class InputPort(AbstractPort):
    
        """
        Input port.
    
        """
    
        _source_signal: Optional[Signal]
    
    
        def __init__(self, operation: "Operation", index: int):
            """Construct an InputPort."""
            super().__init__(operation, index)
    
            self._source_signal = None
    
        @property
        def signal_count(self) -> int:
            return 0 if self._source_signal is None else 1
    
    
        def signals(self) -> Sequence[Signal]:
    
            return [] if self._source_signal is None else [self._source_signal]
    
    
        def add_signal(self, signal: Signal) -> None:
    
            if self._source_signal is not None:
                raise ValueError("Cannot add to already connected input port.")
    
            if signal is self._source_signal:
                raise ValueError("Cannot add same signal twice")
    
            self._source_signal = signal
            signal.set_destination(self)
    
    
        def remove_signal(self, signal: Signal) -> None:
    
            if signal is not self._source_signal:
                raise ValueError("Cannot remove signal that is not connected")
    
            self._source_signal = None
    
    
        def clear(self) -> None:
    
            if self._source_signal is not None:
                self.remove_signal(self._source_signal)
    
        @property
        def connected_source(self) -> Optional["OutputPort"]:
    
            """
            Get the output port that is currently connected to this input port,
    
    Oscar Gustafsson's avatar
    Oscar Gustafsson committed
            return (
                None if self._source_signal is None else self._source_signal.source
            )
    
    Oscar Gustafsson's avatar
    Oscar Gustafsson committed
        def connect(
            self, src: SignalSourceProvider, name: Name = Name("")
        ) -> Signal:
    
            """
            Connect the provided signal source to this input port by creating a new signal.
    
            if self._source_signal is not None:
                raise ValueError("Cannot connect already connected input port.")
    
            if isinstance(src, Signal):
                src.set_destination(self)
                return src
    
            # self._source_signal is set by the signal constructor.
    
    Oscar Gustafsson's avatar
    Oscar Gustafsson committed
            return Signal(source=src.source, destination=self, name=Name(name))
    
    
        def __lshift__(self, src: SignalSourceProvider) -> Signal:
    
            """
            Overloads the left shift operator to make it connect the provided
            signal source to this input port. Returns the new signal.
    
    class OutputPort(AbstractPort, SignalSourceProvider):
    
        """
        Output port.
    
    
        May have zero or more signals connected to it.
    
        """
    
        _destination_signals: List[Signal]
    
    
        def __init__(self, operation: "Operation", index: int):
            """Construct an OutputPort."""
            super().__init__(operation, index)
    
            self._destination_signals = []
    
        @property
        def signal_count(self) -> int:
            return len(self._destination_signals)
    
    
        def signals(self) -> Sequence[Signal]:
    
    
        def add_signal(self, signal: Signal) -> None:
    
            if signal in self._destination_signals:
                raise ValueError("Cannot add same signal twice")
    
            self._destination_signals.append(signal)
    
    
        def remove_signal(self, signal: Signal) -> None:
    
            if signal not in self._destination_signals:
                raise ValueError("Cannot remove signal that is not connected")
    
            self._destination_signals.remove(signal)
            signal.remove_source()
    
    
        def clear(self) -> None:
    
            for signal in copy(self._destination_signals):
    
                self.remove_signal(signal)
    
    
        @property
        def source(self) -> "OutputPort":
            return self