Source code for quantify_scheduler.backends.qblox.operations.inline_q1asm

# Repository: https://gitlab.com/quantify-os/quantify-scheduler
# Licensed according to the LICENCE file on the main branch
"""Qblox specific operation which can be used to inject Q1ASM code directly into a Schedule."""

from __future__ import annotations

import inspect
from copy import deepcopy

from quantify_scheduler.backends.types.qblox import OpInfo
from quantify_scheduler.helpers.collections import make_hash
from quantify_scheduler.operations.operation import Operation


[docs] class InlineQ1ASM(Operation): """ Initialize an InlineQ1ASM operation. This method sets up an operation that contains inline Q1ASM code to be injected directly into a Schedule. All comments in the program will be prefixed with an '[inline]' prefix to help identify the inline assembly within the sequencer program. When using safe labels, then all labels included in the input program will get a prefix of 'inj<digits>_'. By default, safe labels are always used. Labels in comments will not be modified. Parameters ---------- program The Q1ASM program to be injected. duration The duration of the operation in seconds. port The port on which the operation is to be executed. clock The clock associated with the operation. waveforms Dictionary containing waveform information, by default None. safe_labels Flag to indicate if safe labels should be used, by default True. Returns ------- None Notes ----- .. warning:: When using safe_labels=False then all labels in the sequencer program are accessible from inside the inline Q1ASM injection, and so can be jumped to or overwritten. Disabling this feature is available for debugging and advanced compilation strategies only. """ def __init__( self, program: str, duration: float, port: str, clock: str, *, waveforms: dict | None = None, safe_labels: bool = True, ) -> None: waveforms = {} if waveforms is None else waveforms super().__init__(name=str(self.__class__.__name__))
[docs] self._name = self.__class__.__name__
[docs] self.program = program
[docs] self._duration = duration
[docs] self.port = port
[docs] self.clock = clock
[docs] self.waveforms = waveforms
[docs] self.safe_labels = safe_labels
self._update() @property
[docs] def name(self) -> str: """Return the name of the operation.""" return self._name
[docs] def _update(self) -> None: """Update the Operation's internals.""" # No need in this class pass
[docs] def get_used_port_clocks(self) -> set[tuple[str, str]]: """ Extracts which port-clock combinations are used in this operation. Returns ------- : All (port, clock) combinations this operation uses. """ return {(self.port, self.clock)}
def __str__(self) -> str: """ Return a string representation of the InlineQ1ASM operation. This method generates a human-readable string that describes the InlineQ1ASM operation. Returns ------- str A string representation of the InlineQ1ASM operation. """ x = inspect.signature(self.__init__) parameter_list = [] for parameter, _value in x.parameters.items(): cur_atr = self.__getattribute__(parameter) if isinstance(cur_atr, str): cur_atr = f"'{cur_atr}'" parameter_list.append(f"{parameter}={cur_atr}") return f"{self._name}({', '.join(parameter_list)})" # TODO remove once UserDict is removed as superclass def __deepcopy__(self, memo: dict) -> InlineQ1ASM: """Make a deepcopy of this object.""" cls = self.__class__ result = cls.__new__(cls) memo[id(self)] = result for k, v in self.__dict__.items(): setattr(result, k, deepcopy(v, memo)) return result def __hash__(self) -> int: return make_hash( [ self._name, self.program, self.duration, self.port, self.clock, self.waveforms, self.safe_labels, ] )
[docs] class Q1ASMOpInfo(OpInfo): """ Structure describing an inline Q1ASM operation and containing all the information required to play it. """ def __init__(self, inline_q1asm: InlineQ1ASM, operation_start_time: float) -> None: # TODO, make proper attributes when OpInfo is no longer a DataClassJsonMixin info = { "name": inline_q1asm.name, "program": inline_q1asm.program, "duration": inline_q1asm.duration, "port": inline_q1asm.port, "clock": inline_q1asm.clock, "waveforms": inline_q1asm.waveforms, "safe_labels": inline_q1asm.safe_labels, } super().__init__( name=inline_q1asm.name, data=info, timing=operation_start_time, ) def __str__(self) -> str: return f'Assembly "{self.name}" (t0={self.timing}, duration={self.duration})'