Skip to content
Snippets Groups Projects
port.py 3 KiB
Newer Older
  • Learn to ignore specific revisions
  • """@package docstring
    
    B-ASIC Port Module.
    TODO: More info.
    """
    
    from abc import ABC, abstractmethod
    
    from typing import NewType, Optional, List
    
    from b_asic.signal import Signal
    
    Kevin Scott's avatar
    Kevin Scott committed
    from b_asic.operation import Operation
    
    
    PortId = NewType("PortId", int)
    
    
    class Port(ABC):
    
    	"""Abstract port class.
    
    	TODO: More info.
    	"""
    
    
    Kevin Scott's avatar
    Kevin Scott committed
    	_port_id: PortId
    	_operation: Operation
    
    Kevin Scott's avatar
    Kevin Scott committed
    	def __init__(self, port_id: PortId, operation: Operation):
    		self._port_id = port_id
    		self._operation = operation
    
    Kevin Scott's avatar
    Kevin Scott committed
    	@property
    
    	def identifier(self) -> PortId:
    
    		"""Get the unique identifier."""
    
    Kevin Scott's avatar
    Kevin Scott committed
    		return self._port_id
    
    Kevin Scott's avatar
    Kevin Scott committed
    	@property
    	def operation(self) -> Operation:
    		"""Get the connected operation."""
    		return self._operation
    
    	@property
    
    	@abstractmethod
    	def signals(self) -> List[Signal]:
    
    		"""Get a list of all connected signals."""
    
    		pass
    
    Kevin Scott's avatar
    Kevin Scott committed
    	@property
    
    	@abstractmethod
    
    Kevin Scott's avatar
    Kevin Scott committed
    	def signal(self, i: int = 0) -> Signal:
    		"""Get the connected signal at index i."""
    
    		pass
    
    	@abstractmethod
    
    Kevin Scott's avatar
    Kevin Scott committed
    	def signal_count(self) -> int:
    		"""Get the number of connected signals."""
    
    		pass
    
    	@abstractmethod
    	def connect(self, signal: Signal) -> None:
    
    		"""Connect a signal."""
    
    		pass
    
    	@abstractmethod
    	def disconnect(self, i: int = 0) -> None:
    
    		"""Disconnect a signal."""
    
    	# TODO: More stuff.
    
    
    class InputPort(Port):
    
    	"""Input port.
    
    	TODO: More info.
    	"""
    	_source_signal: Optional[Signal]
    
    
    Kevin Scott's avatar
    Kevin Scott committed
    	def __init__(self, port_id: PortId, operation: Operation):
    		super().__init__(port_id, operation)
    
    		self._source_signal = None
    
    
    Kevin Scott's avatar
    Kevin Scott committed
    	@property
    
    	def signals(self) -> List[Signal]:
    
    		return [] if self._source_signal is None else [self._source_signal]
    
    Kevin Scott's avatar
    Kevin Scott committed
    	@property
    
    	def signal(self, i: int = 0) -> Signal:
    
    		assert 0 <= i < self.signal_count() # TODO: Error message.
    		assert self._source_signal is not None # TODO: Error message.
    
    		return self._source_signal
    
    
    Kevin Scott's avatar
    Kevin Scott committed
    	def signal_count(self) -> int:
    		return 0 if self._source_signal is None else 1
    
    
    	def connect(self, signal: Signal) -> None:
    		self._source_signal = signal
    
    Kevin Scott's avatar
    Kevin Scott committed
    		signal.destination = self
    
    
    	def disconnect(self, i: int = 0) -> None:
    
    		assert 0 <= i < self.signal_count() # TODO: Error message.
    
    Kevin Scott's avatar
    Kevin Scott committed
    		self._source_signal.disconnect_source()
    
    		self._source_signal = None
    
    	# TODO: More stuff.
    
    
    class OutputPort(Port):
    
    	"""Output port.
    
    	TODO: More info.
    	"""
    
    	_destination_signals: List[Signal]
    
    
    Kevin Scott's avatar
    Kevin Scott committed
    	def __init__(self, port_id: PortId, operation: Operation):
    		super().__init__(port_id, operation)
    
    		self._destination_signals = []
    
    
    Kevin Scott's avatar
    Kevin Scott committed
    	@property
    
    	def signals(self) -> List[Signal]:
    		return self._destination_signals.copy()
    
    
    Kevin Scott's avatar
    Kevin Scott committed
    	@property
    
    	def signal(self, i: int = 0) -> Signal:
    
    		assert 0 <= i < self.signal_count() # TODO: Error message.
    
    		return self._destination_signals[i]
    
    
    Kevin Scott's avatar
    Kevin Scott committed
    	def signal_count(self) -> int:
    		return len(self._destination_signals)
    
    
    	def connect(self, signal: Signal) -> None:
    		assert signal not in self._destination_signals # TODO: Error message.
    		self._destination_signals.append(signal)
    
    Kevin Scott's avatar
    Kevin Scott committed
    		signal.source = self
    
    	def disconnect(self, i: int = 0) -> None:
    
    		assert 0 <= i < self.signal_count() # TODO: Error message.
    
    		del self._destination_signals[i]