Control flow#
Complex schedules can be constructed from pulses, gates and schedules using control flow. The ConditionalOperation
can be added to a Schedule, or to another control flow operation. Note: ConditionalOperation
cannot be added to another conditional, because currently nested conditionals are not supported.
Adding schedules to a schedule (“subschedules”)#
Supported by
Qblox
andZurich Instruments
backends.
A schedule can be added to a schedule just like an operation.
This is useful e.g. to define a custom composite gate:
from quantify_scheduler.operations.gate_library import X, Y90
from quantify_scheduler import Schedule
def hadamard(qubit: str) -> Schedule:
hadamard_sched = Schedule("hadamard")
hadamard_sched.add(X(qubit))
hadamard_sched.add(Y90(qubit))
return hadamard_sched
my_schedule = Schedule("nice_experiment")
my_schedule.add(X("q1"))
my_schedule.add(hadamard("q1"))
{'name': 'd6b5d0cb-58e6-4ff4-8a38-2cfdfbc4916f', 'operation_id': '3373025095274688200', 'timing_constraints': [{'rel_time': 0, 'ref_schedulable': None, 'ref_pt_new': None, 'ref_pt': None}], 'label': 'd6b5d0cb-58e6-4ff4-8a38-2cfdfbc4916f'}
Note: The repetitions
argument of all but the outermost Schedules is ignored. Schedules can be nested arbitrarily. Timing constraints relative to an inner schedule interpret the inner schedule as one continuous operation. It is not possible to use an operation within a subschedule from outside as reference operation.
Repetition loops#
Supported by
Qblox
backend.
The body
of a LoopOperation
will be repeated repetitions
times.
This can be used to efficiently implement sequential averaging without running over the instruction limit of the hardware:
import numpy as np
from typing import Union
from quantify_scheduler.operations.control_flow_library import LoopOperation
from quantify_scheduler.operations.gate_library import Reset, Measure
def t1_sched_sequential(
times: Union[np.ndarray, float],
qubit: str,
repetitions: int = 1,
) -> Schedule:
times = np.asarray(times)
times = times.reshape(times.shape or (1,))
schedule = Schedule("T1")
for i, tau in enumerate(times):
inner = Schedule(f"inner_{i}")
inner.add(Reset(qubit), label=f"Reset {i}")
inner.add(X(qubit), label=f"pi {i}")
inner.add(
Measure(qubit, acq_index=i),
ref_pt="start",
rel_time=tau,
label=f"Measurement {i}",
)
schedule.add(LoopOperation(body=inner, repetitions=repetitions))
return schedule
Hardware averaging works as expected. In BinMode.APPEND
binning mode, the data is returned in chronological order.
Note
Loops are an experimental feature and come with several limitations at this time, see below.
Limitations#
The time order for zero-duration assembly instructions with the same timing may be incorrect, so verify the compiled schedule (via the generated assembly code). Using loops to implement sequential averaging for qubit spectroscopy is verified to work as expected. Known issues occur in using
SetClockFrequency
andSquarePulse
with duration > 1us at the beginning or end of a loop, for example:
from quantify_scheduler.operations.pulse_library import SquarePulse
schedule = Schedule("T1")
schedule.add(
LoopOperation(
body=SquarePulse(
amp=0.3,
port="q0:res",
duration=2e-6,
clock="q0.ro",
),
repetitions=3
)
)
{'name': '87100100-d67e-4ccd-9784-8d6dc1620f71', 'operation_id': '8279998714109430544', 'timing_constraints': [{'rel_time': 0, 'ref_schedulable': None, 'ref_pt_new': None, 'ref_pt': None}], 'label': '87100100-d67e-4ccd-9784-8d6dc1620f71'}
Repetition loops act on all port-clock combinations present in the circuit. This means that both
X("q0")
andY90("q1")
in the following circuit are repeated three times:
schedule = Schedule("T1")
x = schedule.add(LoopOperation(body=X("q0"), repetitions=3))
schedule.add(Y90("q1"), ref_op=x, ref_pt="start", rel_time=0)
{'name': '81b11757-8aa3-4e79-9f4a-8a8254098e98', 'operation_id': '4871254714119805168', 'timing_constraints': [{'rel_time': 0, 'ref_schedulable': '7c6c3856-7048-46a1-97c5-5e152290b8cd', 'ref_pt_new': None, 'ref_pt': 'start'}], 'label': '81b11757-8aa3-4e79-9f4a-8a8254098e98'}
Safe use with the limitations#
To avoid the limitations mentioned above, it is strongly recommended to use loops only with subschedules, with no operations overlapping with the subschedule. Adding wait times before and after loops ensures that everything works as expected:
from quantify_scheduler.operations.pulse_library import IdlePulse, SquarePulse
inner_schedule = Schedule("inner")
inner_schedule.add(IdlePulse(16e-9))
# anything can go here
inner_schedule.add(
SquarePulse(
amp=0.3,
port="q0:res",
duration=2e-6,
clock="q0.ro",
)
)
# End the inner schedule with a wait time
inner_schedule.add(IdlePulse(16e-9))
outer_schedule = Schedule("outer")
# anything can go here
outer_schedule.add(IdlePulse(16e-9))
outer_schedule.add(LoopOperation(body=inner_schedule, repetitions=5))
outer_schedule.add(IdlePulse(16e-9))
# anything can go here
{'name': 'b0aa77be-847a-4b89-a45e-57fef9a3b8e0', 'operation_id': '3448450716118243670', 'timing_constraints': [{'rel_time': 0, 'ref_schedulable': None, 'ref_pt_new': None, 'ref_pt': None}], 'label': 'b0aa77be-847a-4b89-a45e-57fef9a3b8e0'}