Source code for hermespy.channel.delay.delay

# -*- coding: utf-8 -*-

from __future__ import annotations
from typing import Generic, Set, TypeVar

import numpy as np
from h5py import Group
from scipy.constants import speed_of_light

from hermespy.core import AntennaMode, ChannelStateInformation, ChannelStateFormat, SignalBlock
from hermespy.tools import amplitude_path_loss
from ..channel import (
    Channel,
    ChannelSample,
    LinkState,
    ChannelSampleHook,
    ChannelRealization,
    InterpolationMode,
)

__author__ = "Jan Adler"
__copyright__ = "Copyright 2024, Barkhausen Institut gGmbH"
__credits__ = ["Jan Adler"]
__license__ = "AGPLv3"
__version__ = "1.4.0"
__maintainer__ = "Jan Adler"
__email__ = "jan.adler@barkhauseninstitut.org"
__status__ = "Prototype"


DCRT = TypeVar("DCRT", bound="DelayChannelRealization")
"""Type of delay channel realization"""


[docs] class DelayChannelSample(ChannelSample): """Sample of a delay channel.""" def __init__( self, delay: float, model_propagation_loss: bool, gain: float, state: LinkState ) -> None: # Initialize base class ChannelSample.__init__(self, state) # Store attributes self.__delay = delay self.__model_propagation_loss = model_propagation_loss self.__gain = gain @property def delay(self) -> float: """Propagation delay in seconds.""" return self.__delay @property def model_propagation_loss(self) -> bool: """Should free space propagation loss be modeled?""" return self.__model_propagation_loss @property def gain(self) -> float: """Linear power gain factor a signal experiences when being propagated over this realization.""" return self.__gain @property def expected_energy_scale(self) -> float: return self.__path_gain(self.carrier_frequency) def __spatial_response(self) -> np.ndarray: receiver_position = self.receiver_state.position transmitter_position = self.transmitter_state.position if np.all(receiver_position == transmitter_position): return np.ones( ( self.receiver_state.antennas.num_receive_antennas, self.transmitter_state.antennas.num_transmit_antennas, ), dtype=np.complex128, ) transmit_response = self.transmitter_state.antennas.cartesian_array_response( self.carrier_frequency, self.receiver_state.position, "global", AntennaMode.TX ) receive_response = self.receiver_state.antennas.cartesian_array_response( self.carrier_frequency, self.transmitter_state.position, "global", AntennaMode.RX ) return receive_response @ transmit_response.T def __path_gain(self, carrier_frequency: float) -> float: """Compute the realization's linear path gain factor. carrier_frequency (float): Central frequency of the propagation in :math:`\\mathrm{Hz}`. Returns: Linear path gain scaling the amplitude of a propagated signal. """ gain_factor = self.gain if self.model_propagation_loss: if carrier_frequency == 0.0: raise RuntimeError( "Transmitting device's carrier frequency may not be zero, disable propagation path loss modeling" ) if self.delay > 0: gain_factor *= amplitude_path_loss(carrier_frequency, self.delay * speed_of_light) return gain_factor def _propagate(self, signal: SignalBlock, interpolation: InterpolationMode) -> SignalBlock: delay_samples = round(self.delay * self.bandwidth) spatial_response = self.__spatial_response() gain_factor = self.__path_gain(self.carrier_frequency) propagated_samples = np.append( np.zeros( (self.receiver_state.antennas.num_receive_antennas, delay_samples), np.complex128 ), gain_factor * spatial_response @ signal, axis=1, ) propagated_signal = SignalBlock(propagated_samples, signal._offset) return propagated_signal
[docs] def state( self, num_samples: int, max_num_taps: int, interpolation_mode: InterpolationMode = InterpolationMode.NEAREST, ) -> ChannelStateInformation: delay_samples = round(self.delay * self.bandwidth) spatial_response = self.__spatial_response() gain_factor = self.__path_gain(self.carrier_frequency) cir = np.zeros( ( self.receiver_state.antennas.num_receive_antennas, self.transmitter_state.antennas.num_transmit_antennas, num_samples, min(1 + delay_samples, max_num_taps), ), dtype=np.complex128, ) if delay_samples < max_num_taps: cir[:, :, :, delay_samples] = gain_factor * spatial_response[:, :, np.newaxis] state = ChannelStateInformation(ChannelStateFormat.IMPULSE_RESPONSE, cir) return state
[docs] class DelayChannelRealization(ChannelRealization[DelayChannelSample]): """Base class for delay channel realizations.""" __model_propagation_loss: bool def __init__( self, model_propagation_loss: bool, sample_hooks: Set[ChannelSampleHook[DelayChannelSample]], gain: float, ) -> None: """ Args: model_propagation_loss (bool): Should free space propagation loss be modeled? gain (float): Linear power gain factor a signal experiences when being propagated over this realization. """ # Initialize base class ChannelRealization.__init__(self, sample_hooks, gain) # Initialize class attributes self.__model_propagation_loss = model_propagation_loss @property def model_propagation_loss(self) -> bool: """Should free-space propagation losses be modeled?""" return self.__model_propagation_loss def _reciprocal_sample( self, sample: DelayChannelSample, state: LinkState ) -> DelayChannelSample: return DelayChannelSample(sample.delay, self.model_propagation_loss, self.gain, state)
[docs] def to_HDF(self, group: Group) -> None: group.attrs["model_propagation_loss"] = self.model_propagation_loss group.attrs["gain"] = self.gain
[docs] class DelayChannelBase(Generic[DCRT], Channel[DCRT, DelayChannelSample]): """Base of delay channel models.""" __model_propagation_loss: bool def __init__(self, model_propagation_loss: bool = True, gain: float = 1.0, **kwargs) -> None: """ Args: model_propagation_loss (bool, optional): Should free space propagation loss be modeled? Enabled by default. gain (float, optional): Linear power gain factor a signal experiences when being propagated over this realization. :math:`1.0` by default. \**kawrgs: :class:`Channel` base class initialization arguments. """ # Initialize base class Channel.__init__(self, gain, **kwargs) # Initialize class attributes self.__model_propagation_loss = model_propagation_loss @property def model_propagation_loss(self) -> bool: """Should free space propagation loss be modeled?""" return self.__model_propagation_loss @model_propagation_loss.setter def model_propagation_loss(self, value: bool) -> None: self.__model_propagation_loss = value