# Repository: https://gitlab.com/quantify-os/quantify-scheduler
# Licensed according to the LICENCE file on the main branch
"""Utility classes for Qblox analog modules."""
from __future__ import annotations
import logging
import math
from abc import ABC, abstractmethod
from copy import deepcopy
from typing import TYPE_CHECKING, Any
import numpy as np
from quantify_scheduler.backends.qblox import (
    constants,
    helpers,
    instrument_compilers,
    q1asm_instructions,
)
from quantify_scheduler.backends.qblox.compiler_abc import (
    ClusterModuleCompiler,
    SequencerCompiler,
)
from quantify_scheduler.backends.qblox.enums import (
    ChannelMode,
    DistortionCorrectionLatencyEnum,
    QbloxFilterConfig,
    QbloxFilterMarkerDelay,
)
from quantify_scheduler.backends.qblox.exceptions import NcoOperationTimingError
from quantify_scheduler.backends.qblox.operation_handling.acquisitions import (
    SquareAcquisitionStrategy,
    WeightedAcquisitionStrategy,
)
from quantify_scheduler.backends.qblox.operation_handling.factory_analog import (
    get_operation_strategy,
)
from quantify_scheduler.backends.qblox.operation_handling.virtual import (
    NcoPhaseShiftStrategy,
    NcoResetClockPhaseStrategy,
    NcoSetClockFrequencyStrategy,
)
from quantify_scheduler.backends.types.qblox import (
    AnalogModuleSettings,
    AnalogSequencerSettings,
    BasebandModuleSettings,
    ComplexChannelDescription,
    ComplexInputGain,
    DigitalChannelDescription,
    InputAttenuation,
    OpInfo,
    OutputAttenuation,
    RealInputGain,
    RFDescription,
    RFModuleSettings,
    StaticAnalogModuleProperties,
)
from quantify_scheduler.enums import BinMode
from quantify_scheduler.operations.pulse_library import SetClockFrequency
if TYPE_CHECKING:
    from collections.abc import Iterator
    from quantify_scheduler.backends.qblox.operation_handling.base import (
        IOperationStrategy,
    )
    from quantify_scheduler.backends.qblox.qasm_program import QASMProgram
    from quantify_scheduler.backends.qblox_backend import (
        _ClusterModuleCompilationConfig,
        _SequencerCompilationConfig,
    )
    from quantify_scheduler.resources import Resource
    from quantify_scheduler.schedules.schedule import AcquisitionMetadata
[docs]
logger = logging.getLogger(__name__) 
logger.setLevel(logging.WARNING)
[docs]
class AnalogSequencerCompiler(SequencerCompiler):
    """
    Class that performs the compilation steps on the sequencer level, for QCM and
    QRM-type modules.
    Parameters
    ----------
    parent
        A reference to the module compiler this sequencer belongs to.
    index
        Index of the sequencer.
    static_hw_properties
        The static properties of the hardware.
        This effectively gathers all the differences between the different modules.
    sequencer_cfg
        The instrument compiler config associated to this device.
    """
    def __init__(
        self,
        parent: AnalogModuleCompiler,
        index: int,
        static_hw_properties: StaticAnalogModuleProperties,
        sequencer_cfg: _SequencerCompilationConfig,
    ) -> None:
        super().__init__(
            parent=parent,
            index=index,
            static_hw_properties=static_hw_properties,
            sequencer_cfg=sequencer_cfg,
        )
[docs]
        self._settings: AnalogSequencerSettings = (  # type: ignore  # (override type)
            AnalogSequencerSettings.initialize_from_compilation_config(
                sequencer_cfg=sequencer_cfg,
                connected_output_indices=static_hw_properties._get_connected_output_indices(
                    sequencer_cfg.channel_name,
                ),
                connected_input_indices=static_hw_properties._get_connected_input_indices(
                    sequencer_cfg.channel_name,
                    sequencer_cfg.channel_name_measure,
                ),
                default_nco_en=static_hw_properties.default_nco_en,
            )
        ) 
[docs]
        self.associated_ext_lo = sequencer_cfg.lo_name 
[docs]
        self.downconverter_freq = (
            sequencer_cfg.hardware_description.downconverter_freq
            if isinstance(sequencer_cfg.hardware_description, ComplexChannelDescription)
            else None
        ) 
[docs]
        self.mix_lo = (
            sequencer_cfg.hardware_description.mix_lo
            if not isinstance(sequencer_cfg.hardware_description, DigitalChannelDescription)
            else None
        ) 
[docs]
        self._marker_debug_mode_enable = (
            sequencer_cfg.hardware_description.marker_debug_mode_enable
            if not isinstance(sequencer_cfg.hardware_description, DigitalChannelDescription)
            else None
        ) 
[docs]
        self._default_marker = 0 
        if static_hw_properties.default_markers:
            self._default_marker = static_hw_properties.default_markers[sequencer_cfg.channel_name]
    @property
[docs]
    def settings(self) -> AnalogSequencerSettings:
        """
        Gives the current settings. Overridden from the parent class for type hinting.
        Returns
        -------
        :
            The settings set to this sequencer.
        """
        return self._settings 
    @property
[docs]
    def frequency(self) -> float | None:
        """
        The frequency used for modulation of the pulses.
        Returns
        -------
        :
            The frequency.
        """
        return self._settings.modulation_freq 
    @frequency.setter
    def frequency(self, freq: float) -> None:
        """
        Assigns a modulation frequency to the sequencer.
        Parameters
        ----------
        freq
            The frequency to be used for modulation.
        Raises
        ------
        ValueError
            Attempting to set the modulation frequency to a new value even though a
            different value has been previously assigned.
        """
        if (
            self._settings.modulation_freq is not None
            and not math.isnan(self._settings.modulation_freq)
            and not math.isclose(self._settings.modulation_freq, freq)
        ):
            raise ValueError(
                f"Attempting to set the modulation frequency of '{self.name}' of "
                f"'{self.parent.name}' to {freq:e}, while it has previously been set "
                f"to {self._settings.modulation_freq:e}."
            )
        self._settings.modulation_freq = freq
[docs]
    def get_operation_strategy(
        self,
        operation_info: OpInfo,
    ) -> IOperationStrategy:
        """
        Determines and instantiates the correct strategy object.
        Parameters
        ----------
        operation_info
            The operation we are building the strategy for.
        Returns
        -------
        :
            The instantiated strategy object.
        """
        return get_operation_strategy(
            operation_info,
            self.settings.channel_name,
            self.parent.instrument_cfg.hardware_description,
        ) 
[docs]
    def _prepare_acq_settings(
        self,
        acquisitions: list[IOperationStrategy],
        acq_metadata: AcquisitionMetadata,
    ) -> None:
        """
        Sets sequencer settings that are specific to certain acquisitions.
        For example for a TTL acquisition strategy.
        Parameters
        ----------
        acquisitions
            List of the acquisitions assigned to this sequencer.
        acq_metadata
            Acquisition metadata.
        """
        if len(acquisitions) == 0:
            # no acquisitions, so we do nothing.
            return
        acquisition_infos: list[OpInfo] = list(map(lambda acq: acq.operation_info, acquisitions))
        if acq_metadata.acq_protocol in ("TriggerCount", "ThresholdedTriggerCount"):
            self._settings.ttl_acq_auto_bin_incr_en = acq_metadata.bin_mode == BinMode.DISTRIBUTION
            if len(self.connected_input_indices) == 1:
                self._settings.ttl_acq_input_select = self.connected_input_indices[0]
            else:
                raise ValueError(
                    f"Please make sure you use a single real input for this "
                    f"portclock combination. "
                    f"Found: {len(self.connected_input_indices)} connected. "
                    f"TTL acquisition does not support multiple inputs."
                    f"Problem occurred for port {self.port} with"
                    f"clock {self.clock}, which corresponds to {self.name} of "
                    f"{self.parent.name}."
                )
        if acq_metadata.acq_protocol in (
            "ThresholdedAcquisition",
            "ThresholdedTriggerCount",
            "WeightedThresholdedAcquisition",
        ):
            # We ignore None because it does not interfere with other acquisitions/conditionals.
            # This is because a LatchReset is used for every thresholded acq with a not-None
            # address.
            address_set = {
                acq.data.get("feedback_trigger_address")
                for acq in acquisition_infos
                if acq.data.get("feedback_trigger_address") is not None
            }
            if len(address_set) == 0:
                self._settings.thresholded_acq_trigger_write_en = False
                self._settings.thresholded_acq_trigger_write_address = None
            else:
                address = self._get_unique_value_or_raise(
                    values=address_set,
                    setting_name="feedback_trigger_address",
                )
                self._settings.thresholded_acq_trigger_write_en = True
                self._settings.thresholded_acq_trigger_write_address = address
        if acq_metadata.acq_protocol in [
            "ThresholdedAcquisition",
            "WeightedThresholdedAcquisition",
        ]:
            self._prepare_thresholded_acquisition_settings(acquisition_infos)
        elif acq_metadata.acq_protocol == "ThresholdedTriggerCount":
            thresh_trg_cnt_metadata = self._get_thresholded_trigger_count_metadata_by_acq_channel(
                acquisitions
            )
            for acq_channel, metadata in thresh_trg_cnt_metadata.items():
                acq_ch_metadata = acq_metadata.acq_channel_metadata_by_acq_channel_name(acq_channel)
                acq_ch_metadata.thresholded_trigger_count = metadata
        self._settings.integration_length_acq = self._get_integration_length_from_acquisitions() 
[docs]
    def _prepare_thresholded_acquisition_settings(self, acquisition_infos: list[OpInfo]) -> None:
        def _verify_param_range(
            param_name: str,
            val: float | None,
            min_value: float,
            max_value: float,
            portclock: tuple[str, str],
        ) -> None:
            if val is None:
                return
            if not min_value <= val <= max_value:
                raise ValueError(
                    f"Attempting to configure {param_name} to {val} for the sequencer "
                    f"specified with portclock '{portclock}' and while the "
                    f"hardware requires it to be between {min_value} and {max_value}."
                )
        acq_rotation = self._get_unique_value_or_raise(
            values=(acq.data["acq_rotation"] for acq in acquisition_infos),
            setting_name="acq_rotation",
        )
        acq_threshold = self._get_unique_value_or_raise(
            values=(acq.data["acq_threshold"] for acq in acquisition_infos),
            setting_name="acq_threshold",
        )
        _verify_param_range(
            param_name="acq_rotation",
            val=acq_rotation,
            min_value=constants.MIN_PHASE_ROTATION_ACQ,
            max_value=constants.MAX_PHASE_ROTATION_ACQ,
            portclock=self.portclock,
        )
        self._settings.thresholded_acq_rotation = acq_rotation
        _verify_param_range(
            param_name="acq_threshold",
            val=acq_threshold,
            min_value=constants.MIN_DISCRETIZATION_THRESHOLD_ACQ,
            max_value=constants.MAX_DISCRETIZATION_THRESHOLD_ACQ,
            portclock=self.portclock,
        )
        integration_length = acquisition_infos[0].data.get("duration", 0.0) * 1e9
        self._settings.thresholded_acq_threshold = acq_threshold * integration_length 
[docs]
    def _get_integration_length_from_acquisitions(self) -> int | None:
        """
        Get the (validated) integration_length sequencer setting.
        Get the duration of all SSB integration acquisitions assigned to this sequencer
        and validate that they are all the same.
        """
        integration_length = None
        for op_strat in self.op_strategies:
            if not isinstance(op_strat, (SquareAcquisitionStrategy, WeightedAcquisitionStrategy)):
                continue
            acq_duration_ns = round(op_strat.operation_info.duration * 1e9)
            if acq_duration_ns % constants.GRID_TIME != 0:
                raise ValueError(
                    "Attempting to perform square acquisition with a duration of "
                    f"{acq_duration_ns} ns. Please ensure the duration is a multiple "
                    f"of {constants.GRID_TIME} ns.\n\nException caused by "
                    f"{op_strat!r}."
                )
            if integration_length is None:
                integration_length = acq_duration_ns
            elif integration_length != acq_duration_ns:
                raise ValueError(
                    f"Attempting to set an integration_length of {acq_duration_ns} "
                    f"ns, while this was previously determined to be "
                    f"{integration_length}. Please check whether all square "
                    "acquisitions in the schedule have the same duration."
                )
        return integration_length 
[docs]
    def prepare(self) -> None:
        """
        Perform necessary operations on this sequencer's data before
        :meth:`~quantify_scheduler.backends.qblox.compiler_abc.SequencerCompiler.compile`
        is called.
        """
        self._update_set_clock_frequency_operations()
        self._assert_enough_time_between_freq_phase_updates(self._get_ordered_operations())
        super().prepare() 
[docs]
    def _update_set_clock_frequency_operations(self) -> None:
        for op_strat in self.op_strategies:
            if op_strat.operation_info.name == SetClockFrequency.__name__:
                op_strat.operation_info.data.update(
                    {
                        "interm_freq_old": self.frequency,
                    }
                ) 
    @staticmethod
[docs]
    def _assert_enough_time_between_freq_phase_updates(
        ordered_op_strategies: list[IOperationStrategy],
    ) -> None:
        """Check whether there is enough time between subsequent frequency or phase updates."""
        def check_timing(op: IOperationStrategy, type_of_op: str, last_update: int | float) -> int:
            this_time = round(op.operation_info.timing * 1e9)
            if (diff := this_time - last_update) < constants.MIN_TIME_BETWEEN_NCO_OPERATIONS:
                raise NcoOperationTimingError(
                    f"Operation {op.operation_info} occurred {diff} ns after the "
                    f"previous {type_of_op} update. "
                    f"The minimum time between {type_of_op} "
                    f"updates must be {constants.MIN_TIME_BETWEEN_OPERATIONS} ns."
                )
            return this_time
        last_freq_update = -np.inf
        last_phase_update = -np.inf
        for op in ordered_op_strategies:
            if isinstance(op, NcoSetClockFrequencyStrategy):
                last_freq_update = check_timing(op, "frequency", last_freq_update)
            if isinstance(op, (NcoPhaseShiftStrategy, NcoResetClockPhaseStrategy)):
                last_phase_update = check_timing(op, "phase", last_phase_update) 
[docs]
    def _write_pre_wait_sync_instructions(self, qasm: QASMProgram) -> None:
        """
        Write instructions to the QASM program that must come before the first wait_sync.
        The duration must be equal for all module types.
        """
        default_marker = 0
        module_options = self.parent.instrument_cfg.hardware_description
        if (
            module_options
            and isinstance(module_options, RFDescription)
            and module_options.rf_output_on
            and isinstance(qasm.static_hw_properties, StaticAnalogModuleProperties)
            and qasm.static_hw_properties.default_markers
            and self.settings.channel_name in qasm.static_hw_properties.default_markers
        ):
            default_marker = qasm.static_hw_properties.default_markers[self.settings.channel_name]
        qasm.emit(
            q1asm_instructions.SET_MARKER,
            default_marker,
            comment=f"set markers to {default_marker} (init)",
        ) 
[docs]
    def _insert_qasm(self, op_strategy: IOperationStrategy, qasm_program: QASMProgram) -> None:
        """
        Get Q1ASM instruction(s) from ``op_strategy`` and insert them into ``qasm_program``.
        Optionally wrap pulses and acquisitions in marker pulses depending on the
        ``marker_debug_mode_enable`` setting.
        """
        if self._marker_debug_mode_enable and (
            op_strategy.operation_info.is_acquisition
            or op_strategy.operation_info.data.get("wf_func") is not None
        ):
            marker = self._decide_markers(op_strategy)
            qasm_program.emit(
                q1asm_instructions.SET_MARKER,
                marker,
                comment=f"set markers to {marker} (debug = True)",
            )
            op_strategy.insert_qasm(qasm_program)
            qasm_program.emit(
                q1asm_instructions.SET_MARKER,
                self._default_marker,
                comment=f"set markers to {self._default_marker} (default, debug = True).",
            )
            qasm_program.emit(
                q1asm_instructions.UPDATE_PARAMETERS,
                constants.MIN_TIME_BETWEEN_OPERATIONS,
            )
            qasm_program.elapsed_time += constants.MIN_TIME_BETWEEN_OPERATIONS
        else:
            op_strategy.insert_qasm(qasm_program) 
[docs]
    def _decide_markers(self, operation: IOperationStrategy) -> int:
        """
        Helper method to decide what markers should be pulled high when enable_marker is
        set to True.  Checks what module and operation are being processed, then builds
        a bit string accordingly.
        Note that with the current quantify structure a sequencer cannot have connected
        inputs and outputs simultaneously.  Therefore, the QRM baseband module pulls
        both input or output markers high when doing an operation, as it is impossible
        during compile time to find out what physical port is being used.
        Parameters
        ----------
        operation
            The operation currently being processed by the sequence.
        Returns
        -------
            A bit string passed on to the set_mrk function of the Q1ASM object.
        """
        instrument_type = self.static_hw_properties.instrument_type
        if instrument_type == "QCM":
            marker_bit = 0
            for output in self.connected_output_indices:
                marker_bit |= 1 << output
            return marker_bit
        elif instrument_type == "QRM":
            return 0b1100 if operation.operation_info.is_acquisition else 0b0011
        # For RF modules, the first two indices correspond to path enable/disable.
        # Therefore, the index of the output is shifted by 2.
        elif instrument_type == "QCM_RF":
            # connected outputs are either 0,1 or 2,3 corresponding to
            # marker bitstrings 0b0100 and 0b1000 respectively
            output = self.connected_output_indices[0] // 2
            return 1 << (output + 2) | self._default_marker
        elif instrument_type == "QRM_RF":
            return 0b1010 if operation.operation_info.is_acquisition else 0b0110
        elif instrument_type == "QRC":
            return 0b1
        raise ValueError(f"Unknown instrument type {instrument_type}") 
 
[docs]
class AnalogModuleCompiler(ClusterModuleCompiler, ABC):
    """
    Base class for QCM and QRM-type modules.
    This class is defined as an abstract base class since the distinctions between the
    different devices are defined in subclasses.
    Effectively, this base class contains the functionality shared by all Qblox
    devices and serves to avoid repeated code between them.
    Parameters
    ----------
    name
        Name of the `QCoDeS` instrument this compiler object corresponds to.
    total_play_time
        Total time execution of the schedule should go on for. This parameter is
        used to ensure that the different devices, potentially with different clock
        rates, can work in a synchronized way when performing multiple executions of
        the schedule.
    instrument_cfg
        The instrument compiler config referring to this device.
    """
[docs]
    _settings: AnalogModuleSettings  # type: ignore 
[docs]
    static_hw_properties: StaticAnalogModuleProperties  # type: ignore 
    def __init__(
        self,
        name: str,
        total_play_time: float,
        instrument_cfg: _ClusterModuleCompilationConfig,
    ) -> None:
        super().__init__(
            name=name,
            total_play_time=total_play_time,
            instrument_cfg=instrument_cfg,
        )
[docs]
        self.sequencers: dict[str, AnalogSequencerCompiler] = {} 
[docs]
    def _construct_sequencer_compiler(
        self, index: int, sequencer_cfg: _SequencerCompilationConfig
    ) -> AnalogSequencerCompiler:
        """Create an instance of :class:`AnalogSequencerCompiler`."""
        return AnalogSequencerCompiler(
            parent=self,
            index=index,
            static_hw_properties=self.static_hw_properties,
            sequencer_cfg=sequencer_cfg,
        ) 
    @abstractmethod
[docs]
    def assign_frequencies(
        self,
        sequencer: AnalogSequencerCompiler,
        external_lo: instrument_compilers.LocalOscillatorCompiler | None,
        clock_frequency: float,
    ) -> None:
        """
        An abstract method that should be overridden. Meant to assign an IF frequency
        to each sequencer, and an LO frequency to each output (if applicable).
        Parameters
        ----------
        sequencer
            Sequencer compiler object whose NCO frequency will be determined and set.
        external_lo
            Optional LO compiler object representing an external LO, whose LO frequency
            will be determined and set.
        clock_frequency
            Clock frequency of the clock assigned to the sequencer compiler.
        """ 
    @abstractmethod
[docs]
    def assign_attenuation(self) -> None:
        """
        An abstract method that should be overridden. Meant to assign
        attenuation settings from the hardware configuration if there is any.
        """ 
[docs]
    def prepare(
        self,
        external_los: dict[str, instrument_compilers.LocalOscillatorCompiler] | None = None,
        schedule_resources: dict[str, Resource] | None = None,
        **kwargs,  # noqa: ARG002 (unused arg necessary to fit signature)
    ) -> None:
        """
        Performs the logic needed before being able to start the compilation. In effect,
        this means assigning the pulses and acquisitions to the sequencers and
        calculating the relevant frequencies in case an external local oscillator is
        used.
        Parameters
        ----------
        external_los
            Optional LO compiler objects representing external LOs, whose LO frequency
            will be determined and set.
        schedule_resources
            Mapping from clock name to clock resource, which contains the clock frequency.
        kwargs:
            Potential keyword arguments for other compiler classes.
        """
        self._configure_input_gains()
        self._configure_mixer_offsets()
        self._configure_hardware_distortion_corrections()
        self._construct_all_sequencer_compilers()
        for seq in self.sequencers.values():
            if (
                schedule_resources is not None
                and seq.clock in schedule_resources
                and ChannelMode.DIGITAL not in seq.settings.channel_name
            ):
                if seq.associated_ext_lo is None or external_los is None:
                    external_lo = None
                else:
                    external_lo = external_los[seq.associated_ext_lo]
                self.assign_frequencies(
                    seq,
                    external_lo,
                    schedule_resources[seq.clock]["freq"],
                )
        self.distribute_data()
        for seq in self.sequencers.values():
            seq.prepare()
        self._ensure_single_scope_mode_acquisition_sequencer()
        self.assign_attenuation() 
[docs]
    def _ensure_single_scope_mode_acquisition_sequencer(self) -> None:
        """
        Raises an error if multiple sequencers use scope mode acquisition,
        because that's not supported by the hardware.
        Also, see
        :func:`~quantify_scheduler.instrument_coordinator.components.qblox._AnalogReadoutComponent._determine_scope_mode_acquisition_sequencer_and_qblox_acq_index`
        which also ensures the program that gets uploaded to the hardware satisfies this
        requirement.
        Raises
        ------
        ValueError
            Multiple sequencers have to perform trace acquisition. This is not
            supported by the hardware.
        """
        def is_scope_acquisition(acquisition: OpInfo) -> bool:
            return acquisition.data["protocol"] == "Trace"
        scope_acq_seq = None
        for seq in self.sequencers.values():
            op_infos = [
                op.operation_info for op in seq.op_strategies if op.operation_info.is_acquisition
            ]
            has_scope = any(map(is_scope_acquisition, op_infos))
            if has_scope:
                if scope_acq_seq is not None:
                    helpers.single_scope_mode_acquisition_raise(
                        sequencer_0=scope_acq_seq,
                        sequencer_1=seq.index,
                        module_name=self.name,
                    )
                scope_acq_seq = seq.index 
 
[docs]
class BasebandModuleCompiler(AnalogModuleCompiler, ABC):
    """
    Abstract class with all the shared functionality between the QRM and QCM baseband
    modules.
    """
[docs]
    _settings_type = BasebandModuleSettings  # To be overwritten by subclasses 
    def __init__(
        self,
        name: str,
        total_play_time: float,
        instrument_cfg: _ClusterModuleCompilationConfig,
    ) -> None:
        super().__init__(
            name=name,
            total_play_time=total_play_time,
            instrument_cfg=instrument_cfg,
        )
[docs]
        self._settings = self._settings_type.extract_settings_from_mapping(instrument_cfg) 
[docs]
    def assign_frequencies(
        self,
        sequencer: AnalogSequencerCompiler,
        external_lo: instrument_compilers.LocalOscillatorCompiler | None,
        clock_frequency: float,
    ) -> None:
        """
        Determines LO/IF frequencies and assigns them, for baseband modules.
        In case of **no** external local oscillator, the NCO is given the same
        frequency as the clock -- unless NCO was permanently disabled via
        `"interm_freq": 0` in the hardware config.
        In case of **an** external local oscillator and `sequencer.mix_lo` is
        ``False``, the LO is given the same frequency as the clock
        (via :func:`.helpers.determine_clock_lo_interm_freqs`).
        Parameters
        ----------
        sequencer
            Sequencer compiler object whose NCO frequency will be determined and set.
        external_lo
            Optional LO compiler object representing an external LO, whose LO frequency
            will be determined and set.
        clock_frequency
            Clock frequency of the clock assigned to the sequencer compiler.
        Raises
        ------
        ValueError
            If the NCO and/or LO frequencies cannot be determined, are invalid, or are
            inconsistent with the clock frequency.
        """
        if external_lo is None:
            if sequencer.settings.nco_en:
                sequencer.frequency = clock_frequency
            # Early return: we do not validate further since there is no way we could
            # retrieve an NCO frequency if it was not already set, and if it was not set
            # then nco_en was set to False.
            return
        # In using external local oscillator, determine clock and LO/IF freqs,
        # and then set LO/IF freqs, and enable NCO (via setter)
        try:
            freqs = helpers.determine_clock_lo_interm_freqs(
                freqs=helpers.Frequencies(
                    clock=clock_frequency,
                    LO=external_lo.frequency,
                    IF=sequencer.frequency,
                ),
                downconverter_freq=sequencer.downconverter_freq,
                mix_lo=sequencer.mix_lo,
            )
        except Exception as error:  # Adding sequencer info to exception message
            raise error.__class__(
                f"{error} (for '{sequencer.name}' of '{self.name}' "
                f"with port '{sequencer.port}' and clock '{sequencer.clock}')"
            ) from None
        external_lo.frequency = freqs.LO
        sequencer.frequency = freqs.IF 
[docs]
    def assign_attenuation(self) -> None:
        """
        Meant to assign attenuation settings from the hardware configuration, if there
        is any. For baseband modules there is no attenuation parameters currently.
        """ 
 
[docs]
class RFModuleCompiler(AnalogModuleCompiler, ABC):
    """
    Abstract class with all the shared functionality between the QRM-RF and QCM-RF
    modules.
    """
    def __init__(
        self,
        name: str,
        total_play_time: float,
        instrument_cfg: _ClusterModuleCompilationConfig,
    ) -> None:
        super().__init__(
            name=name,
            total_play_time=total_play_time,
            instrument_cfg=instrument_cfg,
        )
[docs]
        self._settings: RFModuleSettings = (  # type: ignore
            RFModuleSettings.extract_settings_from_mapping(instrument_cfg)
        ) 
[docs]
    def assign_frequencies(
        self,
        sequencer: AnalogSequencerCompiler,
        external_lo: instrument_compilers.LocalOscillatorCompiler | None,  # noqa: ARG002
        clock_frequency: float,
    ) -> None:
        """
        Determines LO/IF frequencies and assigns them for RF modules.
        Parameters
        ----------
        sequencer
            Sequencer compiler object whose NCO frequency will be determined and set.
        external_lo
            Optional LO compiler object representing an external LO. Not used for RF
            modules, since they use the LO frequency in the module settings.
        clock_frequency
            Clock frequency of the clock assigned to the sequencer compiler.
        Raises
        ------
        ValueError
            If the NCO and/or LO frequencies cannot be determined, are invalid, or are
            inconsistent with the clock frequency.
        """
        for lo_idx in RFModuleCompiler._get_connected_lo_indices(sequencer):
            try:
                freqs = helpers.determine_clock_lo_interm_freqs(
                    freqs=helpers.Frequencies(
                        clock=clock_frequency,
                        LO=self._get_lo_frequency(lo_idx),
                        IF=sequencer.frequency,
                    ),
                    downconverter_freq=sequencer.downconverter_freq,
                    mix_lo=True,
                )
            except Exception as error:  # Adding sequencer info to exception message
                raise error.__class__(
                    f"{error} (for '{sequencer.name}' of '{self.name}' "
                    f"with port '{sequencer.port}' and clock '{sequencer.clock}')"
                ) from None
            self._set_lo_frequency(lo_idx, freqs.LO)
            # Calling the frequency setter inside the for-loop helps catch bugs where
            # two different frequencies could accidentally be set on the same sequencer.
            sequencer.frequency = freqs.IF 
    @staticmethod
[docs]
    def _get_connected_lo_indices(sequencer: AnalogSequencerCompiler) -> Iterator[int]:
        """
        Identify the LO the sequencer is outputting.
        Use the sequencer output to module output correspondence, and then
        use the fact that LOX is connected to module output X.
        """
        for sequencer_output_index in sequencer.connected_output_indices:
            if sequencer_output_index % 2 == 0:
                # We will only use real output 0, 2, 4, 6, ... (if they exist) as they
                # are part of the same complex outputs as real output 1, 3, 5, ... in order.
                yield sequencer_output_index // 2 
[docs]
    def _validate_lo_index(self, lo_idx: int) -> None:
        instrument_type = self.static_hw_properties.instrument_type
        if not (
            (instrument_type == "QRM_RF" and lo_idx in (0,))
            or (instrument_type == "QCM_RF" and lo_idx in (0, 1))
            or (instrument_type == "QRC" and lo_idx in range(6))
        ):
            raise IndexError(
                f"Module {self.name} of type "
                f"{self.static_hw_properties.instrument_type} does not have an LO "
                f"index {lo_idx}."
            ) 
[docs]
    def _get_lo_frequency(self, lo_idx: int) -> float | None:
        """
        Get the LO frequency from the settings.
        Parameters
        ----------
        lo_idx : int
            The LO index.
        Returns
        -------
        float | None
            The frequency, or None if it has not been determined yet.
        Raises
        ------
        IndexError
            If the derived class instance does not contain an LO with that index.
        """
        self._validate_lo_index(lo_idx)
        return getattr(self._settings, f"lo{lo_idx}_freq") 
[docs]
    def _set_lo_frequency(self, lo_idx: int, frequency: float) -> None:
        """
        Set the LO frequency from the settings.
        Parameters
        ----------
        lo_idx : int
            The LO index.
        frequency : float
            The frequency.
        Raises
        ------
        IndexError
            If the derived class instance does not contain an LO with that index.
        """
        previous_lo_freq = self._get_lo_frequency(lo_idx)
        if (
            previous_lo_freq is not None
            and not math.isnan(previous_lo_freq)
            and not math.isclose(frequency, previous_lo_freq)
        ):
            raise ValueError(
                f"Attempting to set 'lo{lo_idx}' to frequency "
                f"'{frequency:e}', while it has previously already been set to "
                f"'{previous_lo_freq:e}'!"
            )
        self._validate_lo_index(lo_idx)
        setattr(self._settings, f"lo{lo_idx}_freq", frequency) 
[docs]
    def assign_attenuation(self) -> None:
        """
        Assigns attenuation settings from the hardware configuration.
        Floats that are a multiple of 1 are converted to ints.
        This is needed because the :func:`quantify_core.measurement.control.grid_setpoints`
        converts setpoints to floats when using an attenuation as settable.
        """
        def _convert_to_int(
            value: InputAttenuation | OutputAttenuation | None, label: str
        ) -> int | None:
            if value is None:
                return None
            if not math.isclose(value % 1, 0):
                raise ValueError(f'Trying to set "{label}" to non-integer value {value}')
            return int(value)
        in0_att = out0_att = out1_att = None
        input_att_cfg = self.instrument_cfg.hardware_options.input_att
        output_att_cfg = self.instrument_cfg.hardware_options.output_att
        for (
            portclock,
            path,
        ) in self.instrument_cfg.portclock_to_path.items():
            if path.channel_name == "complex_input_0" and input_att_cfg is not None:
                in0_att = input_att_cfg.get(portclock)
        for (
            portclock,
            path,
        ) in self.instrument_cfg.portclock_to_path.items():
            if path.channel_name == "complex_output_0":
                if input_att_cfg is not None:
                    in0_att_from_output = input_att_cfg.get(portclock)
                    if in0_att_from_output is not None:
                        if in0_att is not None:
                            raise ValueError(
                                f"'input_att' is defined for both 'complex_input_0' and "
                                f"'complex_output_0' on module '{self.name}', which is prohibited. "
                                f"Make sure you define it at a single place."
                            )
                        in0_att = in0_att_from_output
                if output_att_cfg is not None:
                    out0_att = output_att_cfg.get(portclock)
            if path.channel_name == "complex_output_1" and output_att_cfg is not None:
                out1_att = output_att_cfg.get(portclock)
        self._settings.in0_att = _convert_to_int(in0_att, label="in0_att")
        self._settings.out0_att = _convert_to_int(out0_att, label="out0_att")
        self._settings.out1_att = _convert_to_int(out1_att, label="out1_att")