compiler_abc ============ .. py:module:: quantify_scheduler.backends.qblox.compiler_abc .. autoapi-nested-parse:: Compiler base and utility classes for Qblox backend. Module Contents --------------- Classes ~~~~~~~ .. autoapisummary:: quantify_scheduler.backends.qblox.compiler_abc.InstrumentCompiler quantify_scheduler.backends.qblox.compiler_abc.Sequencer quantify_scheduler.backends.qblox.compiler_abc.ClusterModuleCompiler quantify_scheduler.backends.qblox.compiler_abc.BasebandModuleCompiler quantify_scheduler.backends.qblox.compiler_abc.RFModuleCompiler Attributes ~~~~~~~~~~ .. autoapisummary:: quantify_scheduler.backends.qblox.compiler_abc.logger .. py:data:: logger .. py:class:: InstrumentCompiler(parent, name: str, total_play_time: float, instrument_cfg: dict[str, Any], latency_corrections: dict[str, float] | None = None) Bases: :py:obj:`abc.ABC` Abstract base class that defines a generic instrument compiler. The subclasses that inherit from this are meant to implement the compilation steps needed to compile the lists of :class:`~quantify_scheduler.backends.types.qblox.OpInfo` representing the pulse and acquisition information to device-specific instructions. Each device that needs to be part of the compilation process requires an associated ``InstrumentCompiler``. :param parent: Reference to the parent object. :type parent: :class:`~quantify_scheduler.backends.qblox.compiler_container.CompilerContainer` :param name: Name of the `QCoDeS` instrument this compiler object corresponds to. :param 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. :param instrument_cfg: The part of the hardware configuration dictionary referring to this device. This is one of the inner dictionaries of the overall hardware config. :param latency_corrections: Dict containing the delays for each port-clock combination. This is specified in the top layer of hardware config. .. py:method:: prepare() -> None Method that can be overridden to implement logic before the main compilation starts. This step is to extract all settings for the devices that are dependent on settings of other devices. This step happens after instantiation of the compiler object but before the start of the main compilation. .. py:method:: compile(debug_mode: bool, repetitions: int) -> Any :abstractmethod: An abstract method that should be overridden in a subclass to implement the actual compilation. It should turn the pulses and acquisitions added to the device into device-specific instructions. :param debug_mode: Debug mode can modify the compilation process, so that debugging of the compilation process is easier. :param repetitions: Number of times execution of the schedule is repeated. :returns: A data structure representing the compiled program. The type is dependent on implementation. .. py:exception:: NcoOperationTimingError Bases: :py:obj:`ValueError` Exception thrown if there are timing errors for NCO operations. .. py:class:: Sequencer(parent: ClusterModuleCompiler, index: int, portclock: tuple[str, str], static_hw_properties: quantify_scheduler.backends.types.qblox.StaticHardwareProperties, settings: quantify_scheduler.backends.types.qblox.SequencerSettings, latency_corrections: dict[str, float], qasm_hook_func: Callable | None = None, lo_name: str | None = None, downconverter_freq: float | None = None, mix_lo: bool = True, marker_debug_mode_enable: bool = False) Class that performs the compilation steps on the sequencer level. :param parent: A reference to the module compiler this sequencer belongs to. :param index: Index of the sequencer. :param portclock: Tuple that specifies the unique port and clock combination for this sequencer. The first value is the port, second is the clock. :param static_hw_properties: The static properties of the hardware. This effectively gathers all the differences between the different modules. :param settings: The settings set to this sequencer. :param latency_corrections: Dict containing the delays for each port-clock combination. :param qasm_hook_func: Allows the user to inject custom Q1ASM code into the compilation, just prior to returning the final string. :param lo_name: The name of the local oscillator instrument connected to the same output via an IQ mixer. This is used for frequency calculations. :param downconverter_freq: .. warning:: Using ``downconverter_freq`` requires custom Qblox hardware, do not use otherwise. Frequency of the external downconverter if one is being used. Defaults to ``None``, in which case the downconverter is inactive. :param mix_lo: Boolean flag for IQ mixing with LO. Defaults to ``True`` meaning IQ mixing is applied. :param marker_debug_mode_enable: Boolean flag to indicate if markers should be pulled high at the start of operations. Defaults to False, which means the markers will not be used during the sequence. .. py:class:: ParseOperationStatus Bases: :py:obj:`enum.Enum` Return status of the stack. .. py:attribute:: COMPLETED_ITERATION The iterator containing operations is exhausted. .. py:attribute:: EXITED_CONTROL_FLOW The end of a control flow scope is reached. .. py:property:: connected_output_indices :type: tuple[int, Ellipsis] Return the connected output indices associated with the output name specified in the hardware config. For the baseband modules, output index 'n' corresponds to physical module output 'n+1'. For RF modules, output indices '0' and '1' (or: '2' and '3') correspond to 'path_I' and 'path_Q' of some sequencer, and both these paths are routed to the **same** physical module output '1' (or: '2'). .. py:property:: connected_input_indices :type: tuple[int, Ellipsis] Return the connected input indices associated with the input name specified in the hardware config. For the baseband modules, input index 'n' corresponds to physical module input 'n+1'. For RF modules, input indices '0' and '1' correspond to 'path_I' and 'path_Q' of some sequencer, and both paths are connected to physical module input '1'. .. py:property:: portclock :type: tuple[str, str] A tuple containing the unique port and clock combination for this sequencer. :returns: The portclock. .. py:property:: settings :type: quantify_scheduler.backends.types.qblox.SequencerSettings Gives the current settings. :returns: The settings set to this sequencer. .. py:property:: name :type: str The name assigned to this specific sequencer. :returns: The name. .. py:property:: has_data :type: bool Whether or not the sequencer has any data (meaning pulses or acquisitions) assigned to it or not. :returns: Has data been assigned to this sequencer? .. py:property:: frequency :type: float | None The frequency used for modulation of the pulses. :returns: The frequency. .. py:attribute:: qasm_hook_func Allows the user to inject custom Q1ASM code into the compilation, just prior to returning the final string. .. py:attribute:: latency_correction :type: float Latency correction accounted for by delaying the start of the program. .. py:method:: _generate_awg_dict() -> dict[str, Any] Generates the dictionary that contains the awg waveforms in the format accepted by the driver. .. rubric:: Notes The final dictionary to be included in the json that is uploaded to the module is of the form: .. code-block:: program awg waveform_name data index acq waveform_name data index This function generates the awg dictionary. :returns: The awg dictionary. :raises ValueError: I or Q amplitude is being set outside of maximum range. :raises RuntimeError: When the total waveform size specified for a port-clock combination exceeds the waveform sample limit of the hardware. .. py:method:: _generate_weights_dict() -> dict[str, Any] Generates the dictionary that corresponds that contains the acq weights waveforms in the format accepted by the driver. .. rubric:: Notes The final dictionary to be included in the json that is uploaded to the module is of the form: .. code-block:: program awg waveform_name data index acq waveform_name data index This function generates the acq dictionary. :returns: The acq dictionary. :raises NotImplementedError: Currently, only two one dimensional waveforms can be used as acquisition weights. This exception is raised when either or both waveforms contain both a real and imaginary part. .. py:method:: _validate_awg_dict(wf_dict: dict[str, Any]) -> None .. py:method:: _prepare_acq_settings(acquisitions: list[quantify_scheduler.backends.qblox.operation_handling.base.IOperationStrategy], acq_metadata: quantify_scheduler.schedules.schedule.AcquisitionMetadata) -> None Sets sequencer settings that are specific to certain acquisitions. For example for a TTL acquisition strategy. :param acquisitions: List of the acquisitions assigned to this sequencer. :param acq_metadata: Acquisition metadata. .. py:method:: _get_integration_length_from_acquisitions() -> 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. .. py:method:: _generate_acq_declaration_dict(repetitions: int, acq_metadata: quantify_scheduler.schedules.schedule.AcquisitionMetadata) -> dict[str, Any] Generates the "acquisitions" entry of the program json. It contains declaration of the acquisitions along with the number of bins and the corresponding index. For the name of the acquisition (in the hardware), the acquisition channel (cast to str) is used, and is thus identical to the index. Number of bins is taken to be the highest acq_index specified for that channel. :param repetitions: The number of times to repeat execution of the schedule. :param acq_metadata: Acquisition metadata. :returns: The "acquisitions" entry of the program json as a dict. The keys correspond to the names of the acquisitions (i.e. the acq_channel in the scheduler). .. py:method:: generate_qasm_program(total_sequence_time: float, align_qasm_fields: bool, acq_metadata: quantify_scheduler.schedules.schedule.AcquisitionMetadata | None, repetitions: int) -> str Generates a QASM program for a sequencer. Requires the awg and acq dicts to already have been generated. Example of a program generated by this function: .. code-block:: wait_sync 4 set_mrk 1 move 10,R0 # iterator for loop with label start start: wait 4 set_awg_gain 22663,10206 # setting gain for 9056793381316377208 play 0,1,4 wait 176 loop R0,@start set_mrk 0 upd_param 4 stop :param total_sequence_time: Total time the program needs to play for. If the sequencer would be done before this time, a wait is added at the end to ensure synchronization. :param align_qasm_fields: If True, make QASM program more human-readable by aligning its fields. :param acq_metadata: Acquisition metadata. :param repetitions: Number of times to repeat execution of the schedule. :returns: The generated QASM program. :Warns: **RuntimeWarning** -- When number of instructions in the generated QASM program exceeds the maximum supported number of instructions for sequencers in the type of module. :raises RuntimeError: Upon ``total_sequence_time`` exceeding :attr:`.QASMProgram.elapsed_time`. .. py:method:: _parse_operations(operations_iter: Iterator[quantify_scheduler.backends.qblox.operation_handling.base.IOperationStrategy], qasm: quantify_scheduler.backends.qblox.qasm_program.QASMProgram, acquisition_multiplier: int) -> ParseOperationStatus Handle control flow and insert Q1ASM. .. py:method:: _insert_qasm_marker_debug_wrapped(operation: quantify_scheduler.backends.qblox.operation_handling.base.IOperationStrategy, qasm: quantify_scheduler.backends.qblox.qasm_program.QASMProgram) -> None .. py:method:: _initialize_append_mode_registers(qasm: quantify_scheduler.backends.qblox.qasm_program.QASMProgram, op_strategies: list[quantify_scheduler.backends.qblox.operation_handling.acquisitions.AcquisitionStrategyPartial]) -> None Adds the instructions to initialize the registers needed to use the append bin mode to the program. This should be added in the header. :param qasm: The program to add the instructions to. :param op_strategies: An operations list including all the acquisitions to consider. .. py:method:: _get_latency_correction_ns(latency_correction: float) -> int .. py:method:: _insert_update_parameters() -> None Insert update parameter instructions to activate offsets, if they are not already activated by a play, acquire or acquire_weighed instruction (see also `the Q1ASM reference `_). .. py:method:: _any_other_updating_instruction_at_timing_for_parameter_instruction(op_index: int, sorted_pulses_and_acqs: list[quantify_scheduler.backends.qblox.operation_handling.base.IOperationStrategy]) -> bool :staticmethod: .. py:method:: _get_new_update_parameters(pulses_and_acqs: list[quantify_scheduler.backends.qblox.operation_handling.base.IOperationStrategy]) -> list[quantify_scheduler.backends.qblox.operation_handling.base.IOperationStrategy] .. py:method:: _generate_waveforms_and_program_dict(program: str, waveforms_dict: dict[str, Any], weights_dict: dict[str, Any] | None = None, acq_decl_dict: dict[str, Any] | None = None) -> dict[str, Any] :staticmethod: Generates the full waveforms and program dict that is to be uploaded to the sequencer from the program string and the awg and acq dicts, by combining them and assigning the appropriate keys. :param program: The compiled QASM program as a string. :param waveforms_dict: The dictionary containing all the awg data and indices. This is expected to be of the form generated by the ``generate_awg_dict`` method. :param weights_dict: The dictionary containing all the acq data and indices. This is expected to be of the form generated by the ``generate_acq_dict`` method. :param acq_decl_dict: The dictionary containing all the acq declarations. This is expected to be of the form generated by the ``generate_acq_decl_dict`` method. :returns: The combined program. .. py:method:: _dump_waveforms_and_program_json(wf_and_pr_dict: dict[str, Any], label: str | None = None) -> str :staticmethod: Takes a combined waveforms and program dict and dumps it as a json file. :param wf_and_pr_dict: The dict to dump as a json file. :param label: A label that is appended to the filename. :returns: The full absolute path where the json file is stored. .. py:method:: prepare() -> None Perform necessary operations on this sequencer's data before :meth:`~Sequencer.compile` is called. .. py:method:: _update_set_clock_frequency_operations() -> None .. py:method:: compile(sequence_to_file: bool, align_qasm_fields: bool, repetitions: int = 1) -> tuple[dict[str, Any] | None, quantify_scheduler.schedules.schedule.AcquisitionMetadata | None] Performs the full sequencer level compilation based on the assigned data and settings. If no data is assigned to this sequencer, the compilation is skipped and None is returned instead. :param sequence_to_file: Dump waveforms and program dict to JSON file, filename stored in `Sequencer.settings.seq_fn`. :param align_qasm_fields: If True, make QASM program more human-readable by aligning its fields. :param repetitions: Number of times execution the schedule is repeated. :returns: The compiled program and the acquisition metadata. If no data is assigned to this sequencer, the compilation is skipped and None is returned instead. .. py:method:: _replace_marker_pulses(op_strategies: list[quantify_scheduler.backends.qblox.operation_handling.base.IOperationStrategy]) -> list[quantify_scheduler.backends.qblox.operation_handling.base.IOperationStrategy] :staticmethod: Replaces MarkerPulse operations by explicit high and low operations. .. py:method:: _decide_markers(operation) -> 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. :param sequencer: The sequencer currently in the process of constructing a Q1ASM program. :param operation: The operation currently being processed by the sequence. :returns: A bit string passed on to the set_mrk function of the Q1ASM object. .. py:method:: _check_nco_operation_timing(sorted_pulses_and_acqs: list[quantify_scheduler.backends.qblox.operation_handling.base.IOperationStrategy]) -> None :staticmethod: Check whether this sequencer's operation adhere to NCO timing restrictions. .. py:class:: ClusterModuleCompiler(parent, name: str, total_play_time: float, instrument_cfg: dict[str, Any], latency_corrections: dict[str, float] | None = None) Bases: :py:obj:`InstrumentCompiler`, :py:obj:`abc.ABC` Base class for all cluster modules, and an interface for those modules to the :class:`~quantify_scheduler.backends.qblox.instrument_compilers.ClusterCompiler`. 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. :param parent: Reference to the parent object. :type parent: :class:`~quantify_scheduler.backends.qblox.compiler_container.CompilerContainer` :param name: Name of the `QCoDeS` instrument this compiler object corresponds to. :param 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. :param instrument_cfg: The part of the hardware configuration dictionary referring to this device. This is one of the inner dictionaries of the overall hardware config. :param latency_corrections: Dict containing the delays for each port-clock combination. This is specified in the top layer of hardware config. .. py:property:: portclocks :type: list[tuple[str, str]] Returns all the port-clock combinations that this device can target. .. py:property:: supports_acquisition :type: bool :abstractmethod: Specifies whether the device can perform acquisitions. .. py:property:: max_number_of_instructions :type: int :abstractmethod: The maximum number of Q1ASM instructions supported by this module type. .. py:property:: _portclocks_with_data :type: set[tuple[str, str]] All the port-clock combinations associated with at least one pulse and/or acquisition. :returns: A set containing all the port-clock combinations that are used by this InstrumentCompiler. .. py:property:: settings_type :type: type[quantify_scheduler.backends.types.qblox.BaseModuleSettings] :abstractmethod: Specifies the module settings class used by the instrument. .. py:property:: static_hw_properties :type: quantify_scheduler.backends.types.qblox.StaticHardwareProperties :abstractmethod: The static properties of the hardware. This effectively gathers all the differences between the different modules. .. py:method:: add_op_info(port: str, clock: str, op_info: quantify_scheduler.backends.types.qblox.OpInfo) -> None Assigns a certain pulse or acquisition to this device. :param port: The port this waveform is sent to (or acquired from). :param clock: The clock for modulation of the pulse or acquisition. Can be a BasebandClock. :param op_info: Data structure containing all the information regarding this specific pulse or acquisition operation. .. py:method:: _construct_sequencers() -> None Constructs :class:`~Sequencer` objects for each port and clock combination belonging to this device. :raises ValueError: When the output names do not conform to the `complex_output_X`/`real_output_X` norm, where X is the index of the output. :raises KeyError: Raised if no 'portclock_configs' entry is found in the specific outputs of the hardware config. :raises ValueError: Raised when the same port-clock is multiply assigned in the hardware config. :raises ValueError: Attempting to use more sequencers than available. .. py:method:: distribute_data() -> None Distributes the pulses and acquisitions assigned to this module over the different sequencers based on their portclocks. Raises an exception in case the device does not support acquisitions. .. py:method:: assign_frequencies(sequencer: Sequencer) -> None :abstractmethod: 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). .. py:method:: _set_lo_interm_freqs(freqs: quantify_scheduler.backends.qblox.helpers.Frequencies, sequencer: Sequencer, compiler_lo_baseband: quantify_scheduler.backends.qblox.instrument_compilers.LocalOscillatorCompiler | None = None, lo_freq_setting_rf: str | None = None) -> None Sets the LO/IF frequencies, for baseband and RF modules. :param freqs: LO, IF, and clock frequencies, supplied via an :class:`.helpers.Frequencies` object. :param sequencer: The sequencer for which frequences are to be set. :param compiler_lo_baseband: For baseband modules, supply the :class:`.instrument_compilers.LocalOscillatorCompiler` instrument compiler of which the frequency is to be set. :param lo_freq_setting_rf: For RF modules, supply the name of the LO frequency param from the :class:`.RFModuleSettings` that is to be set. :raises ValueError: In case neither LO frequency nor IF has been supplied. :raises ValueError: In case both LO frequency and IF have been supplied and do not adhere to :math:`f_{RF} = f_{LO} + f_{IF}`. :raises ValueError: In case of RF, when the LO frequency was already set to a different value. .. py:method:: assign_attenuation() -> None :abstractmethod: An abstract method that should be overridden. Meant to assign attenuation settings from the hardware configuration if there is any. .. py:method:: prepare() -> 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. .. py:method:: _configure_input_gains() -> None Configures input gain of module settings. Loops through all valid channel names and checks for gain values in hw config. Throws a ValueError if a gain value gets modified. .. py:method:: _configure_mixer_offsets() -> None Configures offset of input, uses calc_from_units_volt found in helper file. Raises an exception if a value outside the accepted voltage range is given. .. py:method:: _ensure_single_scope_mode_acquisition_sequencer() -> 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._QRMComponent._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. .. py:method:: compile(debug_mode: bool, repetitions: int = 1, sequence_to_file: bool | None = None) -> dict[str, Any] Performs the actual compilation steps for this module, by calling the sequencer level compilation functions and combining them into a single dictionary. :param debug_mode: Debug mode can modify the compilation process, so that debugging of the compilation process is easier. :param repetitions: Number of times execution the schedule is repeated. :param sequence_to_file: Dump waveforms and program dict to JSON file, filename stored in `Sequencer.settings.seq_fn`. :returns: The compiled program corresponding to this module. It contains an entry for every sequencer under the key `"sequencers"`, and acquisition metadata under the key `"acq_metadata"`, and the `"repetitions"` is an integer with the number of times the defined schedule is repeated. All the other generic settings are under the key `"settings"`. If the device is not actually used, and an empty program is compiled, None is returned instead. .. py:class:: BasebandModuleCompiler(parent, name: str, total_play_time: float, instrument_cfg: dict[str, Any], latency_corrections: dict[str, float] | None = None) Bases: :py:obj:`ClusterModuleCompiler` Abstract class with all the shared functionality between the QRM and QCM baseband modules. .. py:property:: settings_type :type: type[quantify_scheduler.backends.types.qblox.BasebandModuleSettings] The settings type used by baseband-type devices. .. py:method:: assign_frequencies(sequencer: Sequencer) -> 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`). .. py:method:: assign_attenuation() -> None Meant to assign attenuation settings from the hardware configuration, if there is any. For baseband modules there is no attenuation parameters currently. .. py:class:: RFModuleCompiler(parent, name: str, total_play_time: float, instrument_cfg: dict[str, Any], latency_corrections: dict[str, float] | None = None) Bases: :py:obj:`ClusterModuleCompiler` Abstract class with all the shared functionality between the QRM-RF and QCM-RF modules. .. py:property:: settings_type :type: type[quantify_scheduler.backends.types.qblox.RFModuleSettings] The settings type used by RF modules. .. py:method:: assign_frequencies(sequencer: Sequencer) -> None Determines LO/IF frequencies and assigns them for RF modules. .. py:method:: _get_connected_lo_indices(sequencer: Sequencer) -> Iterator[int] :staticmethod: 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. .. py:method:: assign_attenuation() -> 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.