{ "cells": [ { "cell_type": "markdown", "id": "add17426", "metadata": {}, "source": [ "(sec-acquisitions)=\n", "\n", "\n", "# Tutorial: Acquisitions\n", "\n", "```{seealso}\n", "The complete source code of this tutorial can be found in\n", "\n", "{nb-download}`Acquisitions.ipynb`\n", "```\n", "\n", "## Introduction\n", "\n", "In this tutorial we give examples on how to add acquisitions to schedules, and how to retrieve acquisition results using the {class}`~quantify_scheduler.instrument_coordinator.instrument_coordinator.InstrumentCoordinator`.\n", "More specifically, this tutorial only describes acquisitions with {class}`~quantify_scheduler.instrument_coordinator.instrument_coordinator.InstrumentCoordinator` with Qblox backend with transmon qubits (or Qblox backend with NV center in case of trigger count). See {ref}`sec-tutorial-schedulegettable` for a tutorial on how to perform acquisitions with {class}`~quantify_scheduler.gettables.ScheduleGettable`, and see {ref}`sec-backend-zhinst` for help on how to perform experiments with the Zurich Instruments backend.\n", "\n", "This tutorial assumes you are familiar with compiling schedules and running simple pulses on the Qblox hardware. We also assume, that you have basic familiarity with `xarray` (see [xarray introduction](https://quantify-quantify-core.readthedocs-hosted.com/en/latest/technical_notes/dataset_design/Xarray%20introduction.html) and the [official documentation](https://docs.xarray.dev/en/stable/user-guide/data-structures.html)).\n", "\n", "The basic structure of the returned acquisition data is that it is an {class}`xarray.Dataset`, which consists of multiple {class}`xarray.DataArray`. Each of these {class}`xarray.DataArray`s correspond to one acquisition channel.\n", "\n", "It is important to understand, that this tutorial is Qblox specific with regards to setting up the hardware configuration, and setting up the physical wiring for the hardware, but the schedules and protocols and the return data formats are backend independent.\n", "\n", "### Initial setup\n", "\n", "First, we set up the connection to the cluster and hardware configuration." ] }, { "cell_type": "code", "execution_count": 1, "id": "61fb56c3", "metadata": { "mystnb": { "remove_code_outputs": true } }, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "Data will be saved in:\n", "/home/rsoko/quantify-data\n" ] } ], "source": [ "from quantify_core.data import handling as dh\n", "dh.set_datadir(dh.default_datadir())" ] }, { "cell_type": "markdown", "id": "b24a0566", "metadata": {}, "source": [ "In this tutorial we will use the Qblox dummy device, but for a real hardware, the ip can be provided without the `dummy_cfg` argument." ] }, { "cell_type": "code", "execution_count": 2, "id": "c6046d98", "metadata": {}, "outputs": [], "source": [ "from qblox_instruments import Cluster, ClusterType\n", "from quantify_scheduler.instrument_coordinator.components.qblox import ClusterComponent\n", "\n", "cluster = Cluster(\"cluster0\",\n", " \"\",\n", " dummy_cfg={1: ClusterType.CLUSTER_QRM},\n", " )\n", "cluster_component = ClusterComponent(cluster)" ] }, { "cell_type": "code", "execution_count": 3, "id": "5cc6b030", "metadata": {}, "outputs": [], "source": [ "from quantify_scheduler.device_under_test.quantum_device import QuantumDevice\n", "from quantify_scheduler.device_under_test.transmon_element import BasicTransmonElement\n", "\n", "device = QuantumDevice(\"device\")\n", "transmon0 = BasicTransmonElement(\"q0\")\n", "transmon0.clock_freqs.readout(6e9)\n", "device.add_element(transmon0)\n", "transmon1 = BasicTransmonElement(\"q1\")\n", "transmon1.clock_freqs.readout(6e9)\n", "device.add_element(transmon1)\n", "device.instr_instrument_coordinator(\"instrument_coordinator\")" ] }, { "cell_type": "code", "execution_count": 4, "id": "9ccac547", "metadata": {}, "outputs": [], "source": [ "hardware_config = {\n", " \"backend\": \"quantify_scheduler.backends.qblox_backend.hardware_compile\",\n", " \"cluster0\": {\n", " \"ref\": \"internal\",\n", " \"instrument_type\": \"Cluster\",\n", " \"cluster0_module1\": {\n", " \"instrument_type\": \"QRM\",\n", " \"complex_output_0\": {\n", " \"portclock_configs\": [\n", " {\"port\": \"q0:res\", \"clock\": \"q0.ro\", \"interm_freq\": 0},\n", " {\"port\": \"q1:res\", \"clock\": \"q1.ro\", \"interm_freq\": 0},\n", " ]\n", " }\n", " }\n", " },\n", "}\n", "device.hardware_config(hardware_config)" ] }, { "cell_type": "markdown", "id": "7b4eb65a", "metadata": {}, "source": [ "Note here, we used the internal mixer, so `\"interm_freq\"` was set to `0`.\n", "\n", "We set up the {class}`~quantify_scheduler.instrument_coordinator.instrument_coordinator.InstrumentCoordinator`, and we will use the cluster component." ] }, { "cell_type": "code", "execution_count": 5, "id": "01e887a8", "metadata": {}, "outputs": [], "source": [ "from quantify_scheduler.instrument_coordinator import InstrumentCoordinator\n", "instrument_coordinator = InstrumentCoordinator(\"instrument_coordinator\")" ] }, { "cell_type": "code", "execution_count": 6, "id": "09be6004", "metadata": {}, "outputs": [], "source": [ "instrument_coordinator.add_component(cluster_component)" ] }, { "cell_type": "markdown", "id": "d07c55da", "metadata": {}, "source": [ "## Pulse-level acquisitions\n", "\n", "Pulse-level acquisitions define when and how to store and retrieve input signals coming from the device. They are described by their timing information (start time and length), protocol, `acq_channel` and `acq_index_`.\n", "\n", "* Timing information specifies when the acquisition happens on the schedule,\n", "* protocol defines which common hardware and software formatting is done on the input signal.\n", "* `acq_channel` and `acq_index_` together identify each acquisition for the user in the returned dataset.\n", "\n", "In the following subsections we give examples for each supported acquisition protocol.\n", "\n", "We assume, that these tutorials are run on a Qblox QRM cluster module. On the QRM module {math}`\\text{O}^{[1]}` is connected to {math}`\\text{I}^{[1]}` and {math}`\\text{O}^{[2]}` is connected to {math}`\\text{I}^{[2]}`.\n", "\n", "It takes some time before the sent-out signal appears on the input of the QRM, this is the `time_of_flight`." ] }, { "cell_type": "code", "execution_count": 7, "id": "790ca68c", "metadata": {}, "outputs": [], "source": [ "time_of_flight = 148e-9" ] }, { "cell_type": "markdown", "id": "49b318fd", "metadata": {}, "source": [ "### Trace acquisition\n", "\n", "One of the simplest protocol is the trace (or scope) acquisition protocol. With this protocol, you can retrieve the input signal in very small timesteps (on nanosecond timescale) for a relatively long time (on microsecond timescale). In this subsection we will send out a DRAG pulse on the output, then measure it with the input of the QRM, and plot the retrieved data. The exact duration of the acquisition and sample times depend on the hardware.\n", "\n", "#### Setting up the schedule\n", "\n", "Let's define the duration of the DRAG pulse using the parameter `pulse_duration`." ] }, { "cell_type": "code", "execution_count": 8, "id": "769990da", "metadata": {}, "outputs": [], "source": [ "pulse_duration = 1e-6" ] }, { "cell_type": "markdown", "id": "81ddbdff", "metadata": {}, "source": [ "The schedule is very simple, we transmit the pulse and then we start the trace acquisition which occurs `time_of_flight` seconds after the pulse." ] }, { "cell_type": "code", "execution_count": 9, "id": "661a9834", "metadata": { "mystnb": { "remove_code_outputs": true } }, "outputs": [ { "data": { "text/plain": [ "{'name': '189cbbcf-549f-4af3-9cf8-58711711c6ef', 'operation_repr': \"Trace(duration=1e-06,port='q0:res',clock='q0.ro',acq_channel=0,acq_index=0,bin_mode='average',t0=0)\", 'timing_constraints': [{'rel_time': 1.48e-07, 'ref_schedulable': None, 'ref_pt_new': 'start', 'ref_pt': 'start'}], 'label': '189cbbcf-549f-4af3-9cf8-58711711c6ef'}" ] }, "execution_count": 9, "metadata": {}, "output_type": "execute_result" } ], "source": [ "from quantify_scheduler import Schedule\n", "from quantify_scheduler.operations.pulse_library import IdlePulse, DRAGPulse\n", "from quantify_scheduler.operations.acquisition_library import Trace\n", "\n", "schedule = Schedule(\"trace_acquisition_tutorial\")\n", "\n", "schedule.add(IdlePulse(duration=1e-6))\n", "\n", "schedule.add(\n", " DRAGPulse(\n", " G_amp=0.2, D_amp=0.2, duration=pulse_duration, phase=0,\n", " port=\"q0:res\", clock=\"q0.ro\",\n", " ),\n", ")\n", "schedule.add(\n", " Trace(\n", " duration=pulse_duration,\n", " port=\"q0:res\",\n", " clock=\"q0.ro\",\n", " acq_channel=0,\n", " acq_index=0,\n", " ),\n", " ref_pt=\"start\",\n", " rel_time=time_of_flight\n", ")" ] }, { "cell_type": "code", "execution_count": 10, "id": "0c4f22a1", "metadata": { "tags": [ "remove-cell" ] }, "outputs": [], "source": [ "from qblox_instruments import DummyBinnedAcquisitionData, DummyScopeAcquisitionData\n", "import numpy as np\n", "from quantify_scheduler.waveforms import drag\n", "\n", "def drag_pulse_waveform():\n", " duration=1e-6\n", " t = np.arange(0, duration, 1e-9)\n", " wave = drag(t=t, G_amp=0.1, D_amp=0.1, duration=duration, nr_sigma=4)\n", "\n", " return [ (wave[i].real, wave[i].imag) for i in range(len(t)) ]\n", "\n", "dummy_slot_idx = 1\n", "dummy_scope_acquisition_data = DummyScopeAcquisitionData(\n", " data=drag_pulse_waveform(), out_of_range=(False, False), avg_cnt=(0, 0)\n", ")\n", "\n", "cluster.set_dummy_scope_acquisition_data(\n", " slot_idx=dummy_slot_idx,\n", " sequencer=None,\n", " data=dummy_scope_acquisition_data\n", ")" ] }, { "cell_type": "markdown", "id": "42082d51", "metadata": {}, "source": [ "Let's compile the schedule." ] }, { "cell_type": "code", "execution_count": 11, "id": "b5f4f37c", "metadata": {}, "outputs": [], "source": [ "from quantify_scheduler.backends import SerialCompiler\n", "\n", "compiler = SerialCompiler(name=\"compiler\")\n", "compiled_schedule = compiler.compile(schedule=schedule, config=device.generate_compilation_config())" ] }, { "cell_type": "markdown", "id": "64bbe607", "metadata": {}, "source": [ "#### Running the schedule, retrieving acquisition\n", "\n", "The compiled schedule can be run, and the acquisitions can be retrieved with the {func}`~quantify_scheduler.instrument_coordinator.instrument_coordinator.InstrumentCoordinator.retrieve_acquisition` function." ] }, { "cell_type": "code", "execution_count": 12, "id": "2a8285c0", "metadata": {}, "outputs": [], "source": [ "instrument_coordinator.prepare(compiled_schedule)\n", "instrument_coordinator.start()\n", "instrument_coordinator.wait_done(timeout_sec=10)\n", "\n", "acquisition = instrument_coordinator.retrieve_acquisition()" ] }, { "cell_type": "code", "execution_count": 13, "id": "adbb6039", "metadata": {}, "outputs": [ { "data": { "text/html": [ "
\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "
<xarray.Dataset>\n",
       "Dimensions:        (acq_index_0: 1, trace_index_0: 1000)\n",
       "Coordinates:\n",
       "  * acq_index_0    (acq_index_0) int64 0\n",
       "  * trace_index_0  (trace_index_0) int64 0 1 2 3 4 5 ... 994 995 996 997 998 999\n",
       "Data variables:\n",
       "    0              (acq_index_0, trace_index_0) complex128 0j 0j 0j ... 0j 0j 0j
" ], "text/plain": [ "\n", "Dimensions: (acq_index_0: 1, trace_index_0: 1000)\n", "Coordinates:\n", " * acq_index_0 (acq_index_0) int64 0\n", " * trace_index_0 (trace_index_0) int64 0 1 2 3 4 5 ... 994 995 996 997 998 999\n", "Data variables:\n", " 0 (acq_index_0, trace_index_0) complex128 0j 0j 0j ... 0j 0j 0j" ] }, "execution_count": 13, "metadata": {}, "output_type": "execute_result" } ], "source": [ "acquisition" ] }, { "cell_type": "markdown", "id": "fc4e3560", "metadata": {}, "source": [ "The acquisition data is stored as an {class}`xarray.Dataset`. While it typically consists of multiple {class}`xarray.DataArray`s, this particular dataset contains only one {class}`xarray.DataArray`. This array corresponds to `acq_channel=0` as that was the only acquisition channel we used. Each `acq_index_` value represents a 1 ns measurement, given that the Qblox backend employs a trace acquisition with a granularity of 1 ns. The real and imaginary parts of the data correspond to the I and Q components, respectively.\n", "\n", "We can also plot these results with the following commands. Notice, that because the data is an {class}`xarray.Dataset`, it's very easy to plot and format the data. We only ran the schedule once, so `repetition=0`." ] }, { "cell_type": "code", "execution_count": 14, "id": "061f868f", "metadata": {}, "outputs": [ { "data": { "image/png": "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", "text/plain": [ "
" ] }, "metadata": {}, "output_type": "display_data" } ], "source": [ "import matplotlib.pyplot as plt\n", "\n", "fig, axs = plt.subplots(1)\n", "\n", "acquisition[0].real.plot(ax=axs, label=\"I\")\n", "acquisition[0].imag.plot(ax=axs, label=\"Q\")\n", "\n", "axs.set_title(\"\")\n", "axs.set_ylabel(\"\")\n", "axs.legend()\n", "plt.show()" ] }, { "cell_type": "markdown", "id": "9cbc6656", "metadata": {}, "source": [ "### Single-sideband integration acquisition\n", "\n", "The single-sideband integration protocol involves integrating the complex input signal over a given time period. The integration weight is a square window, and this window's length is the same as the acquisition length. The signal is demodulated using the specified clock before the integration happens.\n", "\n", "In this tutorial, we will send 4 square pulses out, and measure it in 4 separate bins, indexed by all combinations of `acq_channel=0,1` and `acq_index_=0,1`. We will also send out purely real and imaginary pulses (or purely I and purely Q pulses), and observe that they appear as real and imaginary acquisitions. In case of single-sideband integration the integration happens after demodulation.\n", "\n", "\n", "Typically, different acquisition channels are usually set up to refer to different qubits. However, in our simple example, we only use a single qubit port-clock combination for both acquisition channels.\n", "\n", "#### Setting up the schedule\n", "\n", "Let's define how much time the pulse takes with `pulse_duration`." ] }, { "cell_type": "code", "execution_count": 15, "id": "e17ddf99", "metadata": {}, "outputs": [], "source": [ "pulse_duration = 120e-9" ] }, { "cell_type": "markdown", "id": "547b76cd", "metadata": {}, "source": [ "We define a simple helper function that sends out the square pulse with `pulse_level` complex amplitude, and then measures it after `time_of_flight` seconds." ] }, { "cell_type": "code", "execution_count": 16, "id": "04d023ea", "metadata": { "mystnb": { "remove_code_outputs": true } }, "outputs": [], "source": [ "from quantify_scheduler import Schedule\n", "from quantify_scheduler.operations.pulse_library import IdlePulse, SquarePulse\n", "from quantify_scheduler.operations.acquisition_library import SSBIntegrationComplex\n", "from quantify_scheduler.enums import BinMode\n", "\n", "schedule = Schedule(\"ssb_acquisition_tutorial\")\n", "\n", "schedule.add(IdlePulse(duration=1e-6))\n", "\n", "def pulse_and_acquisition(pulse_level, acq_channel, acq_index, schedule, bin_mode=BinMode.AVERAGE):\n", " schedule.add(\n", " SquarePulse(\n", " duration=pulse_duration,\n", " amp=pulse_level,\n", " port=\"q0:res\",\n", " clock=\"q0.ro\",\n", " ),\n", " rel_time=1e-6,\n", " )\n", " schedule.add(\n", " SSBIntegrationComplex(\n", " t0=time_of_flight,\n", " duration=pulse_duration,\n", " port=\"q0:res\",\n", " clock=\"q0.ro\",\n", " acq_channel=acq_channel,\n", " acq_index=acq_index,\n", " bin_mode=bin_mode,\n", " ),\n", " ref_pt=\"start\",\n", " rel_time=time_of_flight\n", " )\n", "\n", "pulse_and_acquisition(pulse_level=0.125, acq_channel=0, acq_index=0, schedule=schedule)\n", "pulse_and_acquisition(pulse_level=0.125j, acq_channel=0, acq_index=1, schedule=schedule)\n", "pulse_and_acquisition(pulse_level=0.25, acq_channel=1, acq_index=0, schedule=schedule)\n", "pulse_and_acquisition(pulse_level=0.25j, acq_channel=1, acq_index=1, schedule=schedule)" ] }, { "cell_type": "markdown", "id": "ca35d15f", "metadata": {}, "source": [ "Notice, that the amplitude is double in case of `acq_channel=1` compared to `acq_channel=0`. Also, the amplitude is complex: in case `acq_index_=0` the amplitude is real, and in case `acq_index_=1` the amplitude is imaginary." ] }, { "cell_type": "code", "execution_count": 17, "id": "29eb2652", "metadata": { "tags": [ "remove-cell" ] }, "outputs": [], "source": [ "from qblox_instruments import DummyBinnedAcquisitionData, DummyScopeAcquisitionData\n", "\n", "dummy_slot_idx = 1\n", "cluster.delete_dummy_binned_acquisition_data(slot_idx=dummy_slot_idx, sequencer=0)\n", "cluster.delete_dummy_binned_acquisition_data(slot_idx=dummy_slot_idx, sequencer=1)\n", "dummy_data_0 = [\n", " DummyBinnedAcquisitionData(data=(16, 0), thres=0, avg_cnt=0),\n", " DummyBinnedAcquisitionData(data=(0, 16), thres=0, avg_cnt=0),\n", "]\n", "cluster.set_dummy_binned_acquisition_data(slot_idx=dummy_slot_idx, sequencer=0, acq_index_name=\"0\", data=dummy_data_0)\n", "dummy_data_1 = [\n", " DummyBinnedAcquisitionData(data=(32, 0), thres=0, avg_cnt=0),\n", " DummyBinnedAcquisitionData(data=(0, 32), thres=0, avg_cnt=0),\n", "]\n", "cluster.set_dummy_binned_acquisition_data(slot_idx=dummy_slot_idx, sequencer=0, acq_index_name=\"1\", data=dummy_data_1)" ] }, { "cell_type": "markdown", "id": "9e11b6f9", "metadata": {}, "source": [ "Let's compile the schedule." ] }, { "cell_type": "code", "execution_count": 18, "id": "006aacb3", "metadata": {}, "outputs": [], "source": [ "from quantify_scheduler.backends import SerialCompiler\n", "\n", "compiler = SerialCompiler(name=\"compiler\")\n", "compiled_schedule = compiler.compile(schedule=schedule, config=device.generate_compilation_config())" ] }, { "cell_type": "markdown", "id": "039a6877", "metadata": {}, "source": [ "#### Running the schedule, retrieving acquisition\n", "\n", "Let's run the schedule, and retrieve the acquisitions." ] }, { "cell_type": "code", "execution_count": 19, "id": "9f906e9a", "metadata": {}, "outputs": [], "source": [ "instrument_coordinator.prepare(compiled_schedule)\n", "instrument_coordinator.start()\n", "instrument_coordinator.wait_done(timeout_sec=10)\n", "\n", "acquisition = instrument_coordinator.retrieve_acquisition()" ] }, { "cell_type": "code", "execution_count": 20, "id": "b598e9c1", "metadata": {}, "outputs": [ { "data": { "text/html": [ "
\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "
<xarray.Dataset>\n",
       "Dimensions:      (acq_index_0: 2, acq_index_1: 2)\n",
       "Coordinates:\n",
       "  * acq_index_0  (acq_index_0) int64 0 1\n",
       "  * acq_index_1  (acq_index_1) int64 0 1\n",
       "Data variables:\n",
       "    0            (acq_index_0) complex128 (0.13333333333333333+0j) 0.13333333...\n",
       "    1            (acq_index_1) complex128 (0.26666666666666666+0j) 0.26666666...
" ], "text/plain": [ "\n", "Dimensions: (acq_index_0: 2, acq_index_1: 2)\n", "Coordinates:\n", " * acq_index_0 (acq_index_0) int64 0 1\n", " * acq_index_1 (acq_index_1) int64 0 1\n", "Data variables:\n", " 0 (acq_index_0) complex128 (0.13333333333333333+0j) 0.13333333...\n", " 1 (acq_index_1) complex128 (0.26666666666666666+0j) 0.26666666..." ] }, "execution_count": 20, "metadata": {}, "output_type": "execute_result" } ], "source": [ "acquisition" ] }, { "cell_type": "markdown", "id": "04ad9786", "metadata": {}, "source": [ "There are now two {class}`xarray.DataArray`s in this {class}`xarray.Dataset`. These correspond to `acq_channel=0` and `acq_channel=1`. Both of these `DataArrays` have the following two dimensions: `acq_index_` and `repetition`. Because the schedule was run only once, `repetition=0` for all values.\n", "\n", "As expected, the single side band integration produced a single complex number in each bin. One purely real value in `acq_index_=0` and one purely imaginary value in `acq_index_=1`. Notice how these values are twice as much for `acq_index_=1` as compared to those for `acq_index_=0`.\n", "\n", "#### Bin modes and repetitions\n", "\n", "`quantify-scheduler` offers two kinds of bin modes, that deal with repeated schedules:\n", "\n", "- Average bin mode: Enables repeated measurements and averaging for reduced errors.\n", "- Append bin mode: Allows repeating measurements and retrieving data for each repetition individually.\n", "\n", "```{note}\n", "`QuantumDevice.cfg_sched_repetitions` has no effect on experiments running with {class}`~quantify_scheduler.instrument_coordinator.instrument_coordinator.InstrumentCoordinator` itself by the user. This parameter has only effect if you're using {class}`~quantify_scheduler.gettables.ScheduleGettable`.\n", "```\n", "\n", "To determine the number of times you want the `quantify-scheduler` to execute the schedule, you can set the `repetitions` argument or attribute for the `Schedule` object. By specifying a value for `repetitions`, you can control the number of times the schedule will run. For example, if you set `repetitions` to `8`, the following code snippet demonstrates a schedule that would execute eight times:\n", "\n", "\n", "```{code-block} python\n", "schedule = Schedule(\"Repeated schedule\", repetitions=8)\n", "```\n", "\n", "```{note}\n", "Important: mixing bin modes is not allowed and all bin modes must be the same for each acquisition in one schedule!\n", "```\n", "\n", "##### Average bin mode\n", "\n", "To specify which bin mode you would like to use, set the `bin_mode` argument for each acquisition operation. By default, they are set to `BinMode.AVERAGE`.\n", "\n", "```{code-block} python\n", "from quantify_scheduler.enums import BinMode\n", "\n", "schedule.add(\n", " SSBIntegrationComplex(\n", " duration=pulse_duration,\n", " port=\"q0:res\",\n", " clock=\"q0.ro\",\n", " acq_channel=acq_channel,\n", " acq_index=acq_index,\n", " bin_mode=BinMode.AVERAGE,\n", " )\n", ")\n", "```\n", "\n", "```{note}\n", "Trace acquisitions only work with average bin mode. Integration type acquisitions can be used with append bin mode too.\n", "```\n", "\n", "##### Append bin mode\n", "\n", "Let's create a schedule which is run 3 times in a row in append mode." ] }, { "cell_type": "code", "execution_count": 21, "id": "84975018", "metadata": {}, "outputs": [], "source": [ "from quantify_scheduler import Schedule\n", "from quantify_scheduler.operations.pulse_library import IdlePulse, SquarePulse\n", "from quantify_scheduler.operations.acquisition_library import SSBIntegrationComplex\n", "from quantify_scheduler.enums import BinMode\n", "\n", "schedule = Schedule(\"append_tutorial\", repetitions=3)\n", "\n", "schedule.add(IdlePulse(duration=1e-6))\n", "\n", "pulse_and_acquisition(pulse_level=0.125, acq_channel=0, acq_index=0, schedule=schedule, bin_mode=BinMode.APPEND)\n", "pulse_and_acquisition(pulse_level=0.25, acq_channel=0, acq_index=1, schedule=schedule, bin_mode=BinMode.APPEND)" ] }, { "cell_type": "code", "execution_count": 22, "id": "9cf6b215", "metadata": { "tags": [ "remove-cell" ] }, "outputs": [], "source": [ "from qblox_instruments import DummyBinnedAcquisitionData, DummyScopeAcquisitionData\n", "\n", "dummy_slot_idx = 1\n", "cluster.delete_dummy_binned_acquisition_data(slot_idx=dummy_slot_idx, sequencer=0)\n", "cluster.delete_dummy_binned_acquisition_data(slot_idx=dummy_slot_idx, sequencer=1)\n", "dummy_data_0 = [\n", " DummyBinnedAcquisitionData(data=(16, 0), thres=0, avg_cnt=0),\n", " DummyBinnedAcquisitionData(data=(32, 0), thres=0, avg_cnt=0),\n", "] * 3\n", "cluster.set_dummy_binned_acquisition_data(slot_idx=dummy_slot_idx, sequencer=0, acq_index_name=\"0\", data=dummy_data_0)" ] }, { "cell_type": "markdown", "id": "e55164c9", "metadata": {}, "source": [ "Let's compile the schedule." ] }, { "cell_type": "code", "execution_count": 23, "id": "8a6c6dce", "metadata": {}, "outputs": [], "source": [ "from quantify_scheduler.backends import SerialCompiler\n", "\n", "compiler = SerialCompiler(name=\"compiler\")\n", "compiled_schedule = compiler.compile(schedule=schedule, config=device.generate_compilation_config())" ] }, { "cell_type": "markdown", "id": "113653fb", "metadata": {}, "source": [ "And retrieve the acquisitions" ] }, { "cell_type": "code", "execution_count": 24, "id": "c3a1a2ac", "metadata": {}, "outputs": [], "source": [ "instrument_coordinator.prepare(compiled_schedule)\n", "instrument_coordinator.start()\n", "instrument_coordinator.wait_done(timeout_sec=10)\n", "\n", "acquisition = instrument_coordinator.retrieve_acquisition()" ] }, { "cell_type": "code", "execution_count": 25, "id": "7e2583c5", "metadata": {}, "outputs": [ { "data": { "text/html": [ "
\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "
<xarray.Dataset>\n",
       "Dimensions:      (acq_index_0: 2, repetition: 3)\n",
       "Coordinates:\n",
       "  * acq_index_0  (acq_index_0) int64 0 1\n",
       "Dimensions without coordinates: repetition\n",
       "Data variables:\n",
       "    0            (repetition, acq_index_0) complex128 (0.13333333333333333+0j...
" ], "text/plain": [ "\n", "Dimensions: (acq_index_0: 2, repetition: 3)\n", "Coordinates:\n", " * acq_index_0 (acq_index_0) int64 0 1\n", "Dimensions without coordinates: repetition\n", "Data variables:\n", " 0 (repetition, acq_index_0) complex128 (0.13333333333333333+0j..." ] }, "execution_count": 25, "metadata": {}, "output_type": "execute_result" } ], "source": [ "acquisition" ] }, { "cell_type": "markdown", "id": "76993c94", "metadata": {}, "source": [ "Notice, that now we have `3*2` acquisition values, as expected: with 3 repetitions, and for each repetition 2 acquisitions. Let's select the values for the second run." ] }, { "cell_type": "code", "execution_count": 26, "id": "2061cec6", "metadata": {}, "outputs": [ { "data": { "text/html": [ "
\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "
<xarray.DataArray 0 (acq_index_0: 2)>\n",
       "array([0.13333333+0.j, 0.26666667+0.j])\n",
       "Coordinates:\n",
       "  * acq_index_0  (acq_index_0) int64 0 1
" ], "text/plain": [ "\n", "array([0.13333333+0.j, 0.26666667+0.j])\n", "Coordinates:\n", " * acq_index_0 (acq_index_0) int64 0 1" ] }, "execution_count": 26, "metadata": {}, "output_type": "execute_result" } ], "source": [ "acquisition[0].sel(repetition=1)" ] }, { "cell_type": "markdown", "id": "a5b2bf5a", "metadata": {}, "source": [ "As expected, it has only two values, and the value of `acq_index_=1` is double that of `acq_index_=0`.\n", "\n", "### Trigger count acquisition\n", "\n", "The trigger count acquisition protocol is used for measuring how many times the input signal goes over some limit. This protocol is used, for example, in the case of an NV center type of qubit, or other types of qubit, where counting the number of photons (indirectly as an electrical signal) is important.\n", "\n", "The Trigger Count protocol in `quantify-scheduler` offers two bin modes: average and append mode. These bin modes function differently compared to the Single Sideband Integration protocol.\n", "\n", "In the append bin mode, the resulting data will be a list of 1s, with the length of the list corresponding to the number of triggers that occurred during the acquisition. For example, if there were three triggers, the result would be a list with three 1s: `[1, 1, 1]`.\n", "\n", "In the average bin mode, the result is a distribution, mapping from the count numbers to the number of occurrences for each count numbers. It provides insights into the overall occurrence of triggers when running the acquisition multiple times. Let's consider an example where we execute a schedule three times:\n", "\n", "- during the 1st run, three triggers are acquired\n", "- during the 2nd run, one trigger is acquired,\n", "- during the 3rd run, one trigger is acquired.\n", "\n", "The overall distribution of triggers would be: 1 trigger occurred twice, and 3 triggers occurred once. Hence, the resulting dictionary would be: `{1: 2, 3: 1}`.\n", "\n", "The dictionary notation shows the number of triggers as keys and their corresponding frequencies as values.\n", "\n", "In this tutorial we will try to see how average bin mode works.\n", "Let's create a schedule which consists of several acquisitions which measure how many times some input pulses occurred with `quantify-scheduler`. In this tutorial we assume input pulses are generated from an external source (we do not generate these sources from the hardware).\n", "\n", "The trigger count protocol is currently only implemented for the Qblox backend.\n", "\n", "\n", "#### Setting up the schedule\n", "\n", "Let's define how long time the pulse takes with `pulse_duration`." ] }, { "cell_type": "code", "execution_count": 27, "id": "2b0115cc", "metadata": {}, "outputs": [], "source": [ "pulse_duration = 120e-9" ] }, { "cell_type": "markdown", "id": "17b13f8d", "metadata": {}, "source": [ "and create the schedule mentioned in the introduction: the hardware will run the trigger count acquisition overall 3 times, therefore `repetitions=3` for the schedule, and in each repetition we run the trigger count acquisition once.\n", "The input signals are the following: the first time the schedule runs there are 3 trigger signals and then 1 trigger signal, and there is again 1 trigger signal at the end." ] }, { "cell_type": "code", "execution_count": 28, "id": "8591ba32", "metadata": { "mystnb": { "remove_code_outputs": true } }, "outputs": [ { "data": { "text/plain": [ "{'name': '8688a844-88f4-4a8c-a40c-39fa4eebc748', 'operation_repr': \"TriggerCount(port='q1:res',clock='q1.ro',duration=1.2e-07,acq_channel=0,acq_index=0,bin_mode='average',t0=1.48e-07)\", 'timing_constraints': [{'rel_time': 0, 'ref_schedulable': None, 'ref_pt_new': 'start', 'ref_pt': 'end'}], 'label': '8688a844-88f4-4a8c-a40c-39fa4eebc748'}" ] }, "execution_count": 28, "metadata": {}, "output_type": "execute_result" } ], "source": [ "from quantify_scheduler import Schedule\n", "from quantify_scheduler.operations.pulse_library import IdlePulse, SquarePulse\n", "from quantify_scheduler.operations.acquisition_library import TriggerCount\n", "from quantify_scheduler.enums import BinMode\n", "\n", "schedule = Schedule(\"trigger_count_acquisition_tutorial\", repetitions=3)\n", "\n", "schedule.add(IdlePulse(duration=1e-6))\n", "\n", "pulse_duration = 120e-9\n", "\n", "schedule.add(\n", " TriggerCount(\n", " t0=time_of_flight,\n", " duration=pulse_duration,\n", " port=\"q1:res\",\n", " clock=\"q1.ro\",\n", " acq_channel=0,\n", " bin_mode=BinMode.AVERAGE,\n", " )\n", ")" ] }, { "cell_type": "markdown", "id": "2ec198db", "metadata": {}, "source": [ "The schedule consists of three parts. First 3 trigger pulses, then an acquisition. Then 1 trigger pulse and another acquisition, and finally again one trigger pulse and another acquisition. Notice, that the acquisitions align almost exactly with the pulses, but they are delayed by the delay time. It's also important, that the acquisition channels are the same for both acquisitions. If all of the acquisition channels were different, we would get 3 distributions (for the three acquisition channels)." ] }, { "cell_type": "code", "execution_count": 29, "id": "b38a79d7", "metadata": { "tags": [ "remove-cell" ] }, "outputs": [], "source": [ "from qblox_instruments import DummyBinnedAcquisitionData, DummyScopeAcquisitionData\n", "\n", "dummy_slot_idx = 1\n", "cluster.delete_dummy_binned_acquisition_data(slot_idx=dummy_slot_idx, sequencer=0)\n", "cluster.delete_dummy_binned_acquisition_data(slot_idx=dummy_slot_idx, sequencer=1)\n", "dummy_data_0 = [\n", " DummyBinnedAcquisitionData(data=(16, 0), thres=0, avg_cnt=3),\n", " DummyBinnedAcquisitionData(data=(16, 0), thres=0, avg_cnt=1),\n", " DummyBinnedAcquisitionData(data=(16, 0), thres=0, avg_cnt=1),\n", "]\n", "cluster.set_dummy_binned_acquisition_data(slot_idx=dummy_slot_idx, sequencer=0, acq_index_name=\"0\", data=dummy_data_0)" ] }, { "cell_type": "markdown", "id": "e7f1d62b", "metadata": {}, "source": [ "Let's compile the schedule." ] }, { "cell_type": "code", "execution_count": 30, "id": "a9c44a3b", "metadata": {}, "outputs": [], "source": [ "from quantify_scheduler.backends import SerialCompiler\n", "\n", "compiler = SerialCompiler(name=\"compiler\")\n", "compiled_schedule = compiler.compile(schedule=schedule, config=device.generate_compilation_config())" ] }, { "cell_type": "markdown", "id": "33891bff", "metadata": {}, "source": [ "#### Running the schedule, retrieving acquisition\n", "\n", "Let's run the schedule, and retrieve the acquisitions." ] }, { "cell_type": "code", "execution_count": 31, "id": "aa7665c8", "metadata": {}, "outputs": [], "source": [ "instrument_coordinator.prepare(compiled_schedule)\n", "instrument_coordinator.start()\n", "instrument_coordinator.wait_done(timeout_sec=10)\n", "\n", "acquisition = instrument_coordinator.retrieve_acquisition()" ] }, { "cell_type": "code", "execution_count": 32, "id": "a6f30169", "metadata": {}, "outputs": [ { "data": { "text/html": [ "
\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "
<xarray.Dataset>\n",
       "Dimensions:     (repetition: 1, counts: 2)\n",
       "Coordinates:\n",
       "  * repetition  (repetition) int64 0\n",
       "  * counts      (counts) int64 1 3\n",
       "Data variables:\n",
       "    0           (repetition, counts) float64 2.0 1.0
" ], "text/plain": [ "\n", "Dimensions: (repetition: 1, counts: 2)\n", "Coordinates:\n", " * repetition (repetition) int64 0\n", " * counts (counts) int64 1 3\n", "Data variables:\n", " 0 (repetition, counts) float64 2.0 1.0" ] }, "execution_count": 32, "metadata": {}, "output_type": "execute_result" } ], "source": [ "acquisition" ] }, { "cell_type": "markdown", "id": "8b0c2db0", "metadata": {}, "source": [ "There were three trigger acquisitions overall. In the first acquisition 3 triggers were sent-out, and in the second and third case, only one. So, we expect to see that one trigger was measured twice, and 3 triggers were measured only once. The data shows exactly this. At `acq_channel=0` (corresponding to the `0` key in the `Dataset`) the values are `2` and `1`, with `counts` for `1` and `3` respectively.\n", "\n", "## Gate-level acquisitions\n", "\n", "In the previous section the schedule was defined on the hardware level, in terms of signals and pulses. In this section we will address the acquisitions in terms of qubits. To do that, first, we need to set up a qubit. See {ref}`sec-tutorial-ops-qubits` for an introduction to how to set up a schedule on the gate-level. Integration type acquisitions and trigger count acquisitions make sense on the gate-level, depending on the physical implementation of your qubit.\n", "\n", "In this tutorial we will set up a simple single sideband integration acquisition on the gate-level. In the case of a transmon qubit, a {class}`~quantify_scheduler.operations.gate_library.Measure` gate first sends out an acquisition pulse, and then acquires the signal. The {class}`~quantify_scheduler.device_under_test.quantum_device.QuantumDevice` stores the parameters of how the measurement gate is translated to device level operations by `quantify-scheduler`.\n", "\n", "Let's see what is the effect of modifying the amplitude of the acquisition pulse on the acquisition result. Let's set up the delay time as before, but now on the {class}`~quantify_scheduler.device_under_test.quantum_device.QuantumDevice`, and set up the amplitude of the acquisition pulse, which is a square pulse in this case." ] }, { "cell_type": "code", "execution_count": 33, "id": "7df2657a", "metadata": { "tags": [ "remove-cell" ] }, "outputs": [], "source": [ "device.remove_element(\"q0\")\n", "transmon0.close()" ] }, { "cell_type": "code", "execution_count": 34, "id": "c8956bd6", "metadata": {}, "outputs": [], "source": [ "time_of_flight = 148e-9\n", "pulse_duration = 120e-9" ] }, { "cell_type": "code", "execution_count": 35, "id": "c7c2490e", "metadata": {}, "outputs": [], "source": [ "transmon0 = BasicTransmonElement(\"q0\")\n", "transmon0.clock_freqs.readout(6e9)\n", "transmon0.measure.pulse_amp(0.125)\n", "transmon0.measure.pulse_duration(pulse_duration)\n", "transmon0.measure.acq_delay(time_of_flight)\n", "transmon0.measure.integration_time(pulse_duration)\n", "transmon0.measure.acq_channel(2)\n", "device.add_element(transmon0)" ] }, { "cell_type": "markdown", "id": "facf9263", "metadata": {}, "source": [ "Similar to the previous setup, the pulse has an amplitude of 0.125 and matches the duration of the acquisition. In this case, `quantify-scheduler` will use the port `\":res\"` for both the acquisition pulse and acquisition itself, specifically `\"q0:res\"`. Note, we set the `acq_channel` to `2` for the sake of the example.\n", "\n", "The relevant hardware configuration is the following.\n", "\n", "```{code-block} python\n", "\"portclock_configs\": [\n", " {\"port\": \"q0:res\", \"clock\": \"q0.ro\", \"interm_freq\": 0},\n", " ]\n", "```\n", "\n", "### Creating and running the schedule\n", "\n", "The qubit is now set up, and we can create the schedule." ] }, { "cell_type": "code", "execution_count": 36, "id": "3341a275", "metadata": { "mystnb": { "remove_code_outputs": true } }, "outputs": [ { "data": { "text/plain": [ "{'name': '8338238b-0c67-4115-9822-9e45907e4933', 'operation_repr': 'Measure(\\'q0\\', acq_index=0, acq_protocol=\"None\", bin_mode=None)', 'timing_constraints': [{'rel_time': 1e-06, 'ref_schedulable': None, 'ref_pt_new': 'start', 'ref_pt': 'end'}], 'label': '8338238b-0c67-4115-9822-9e45907e4933'}" ] }, "execution_count": 36, "metadata": {}, "output_type": "execute_result" } ], "source": [ "from quantify_scheduler import Schedule\n", "from quantify_scheduler.operations.pulse_library import IdlePulse\n", "from quantify_scheduler.operations.gate_library import Measure\n", "\n", "schedule = Schedule(\"gate_level_ssb_acquisition_tutorial\")\n", "\n", "schedule.add(IdlePulse(duration=1e-6))\n", "\n", "schedule.add(\n", " Measure(\"q0\", acq_index=0),\n", " rel_time=1e-6,\n", ")" ] }, { "cell_type": "code", "execution_count": 37, "id": "83690814", "metadata": { "tags": [ "remove-cell" ] }, "outputs": [], "source": [ "from qblox_instruments import DummyBinnedAcquisitionData, DummyScopeAcquisitionData\n", "\n", "dummy_slot_idx = 1\n", "cluster.delete_dummy_binned_acquisition_data(slot_idx=dummy_slot_idx, sequencer=0)\n", "cluster.delete_dummy_binned_acquisition_data(slot_idx=dummy_slot_idx, sequencer=1)\n", "dummy_data_0 = [\n", " DummyBinnedAcquisitionData(data=(16, 0), thres=0, avg_cnt=0),\n", "]\n", "cluster.set_dummy_binned_acquisition_data(slot_idx=dummy_slot_idx, sequencer=0, acq_index_name=\"2\", data=dummy_data_0)" ] }, { "cell_type": "markdown", "id": "70e703bf", "metadata": {}, "source": [ "Let's compile the schedule." ] }, { "cell_type": "code", "execution_count": 38, "id": "6d2bce16", "metadata": {}, "outputs": [], "source": [ "from quantify_scheduler.backends import SerialCompiler\n", "\n", "compiler = SerialCompiler(name=\"compiler\")\n", "compiled_schedule = compiler.compile(schedule=schedule, config=device.generate_compilation_config())" ] }, { "cell_type": "markdown", "id": "523e9b62", "metadata": {}, "source": [ "#### Running the schedule, retrieving acquisition\n", "\n", "Let's run the schedule, and retrieve the acquisitions." ] }, { "cell_type": "code", "execution_count": 39, "id": "9d30b47e", "metadata": {}, "outputs": [], "source": [ "instrument_coordinator.prepare(compiled_schedule)\n", "instrument_coordinator.start()\n", "instrument_coordinator.wait_done(timeout_sec=10)\n", "\n", "acquisition = instrument_coordinator.retrieve_acquisition()" ] }, { "cell_type": "code", "execution_count": 40, "id": "8f20f1ed", "metadata": {}, "outputs": [ { "data": { "text/html": [ "
\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "
<xarray.Dataset>\n",
       "Dimensions:      (acq_index_2: 1)\n",
       "Coordinates:\n",
       "  * acq_index_2  (acq_index_2) int64 0\n",
       "Data variables:\n",
       "    2            (acq_index_2) complex128 (0.13333333333333333+0j)
" ], "text/plain": [ "\n", "Dimensions: (acq_index_2: 1)\n", "Coordinates:\n", " * acq_index_2 (acq_index_2) int64 0\n", "Data variables:\n", " 2 (acq_index_2) complex128 (0.13333333333333333+0j)" ] }, "execution_count": 40, "metadata": {}, "output_type": "execute_result" } ], "source": [ "acquisition" ] }, { "cell_type": "markdown", "id": "54ec080f", "metadata": {}, "source": [ "Notice, that the result is only one number at `acq_channel=2`.\n", "\n", "#### Modifying the readout pulse amplitude\n", "\n", "Let's see what the effect on the measurement is if we double the pulse amplitude of the readout pulse" ] }, { "cell_type": "code", "execution_count": 41, "id": "bc5285df", "metadata": { "tags": [ "remove-cell" ] }, "outputs": [], "source": [ "device.remove_element(\"q0\")" ] }, { "cell_type": "code", "execution_count": 42, "id": "096fbe45", "metadata": {}, "outputs": [], "source": [ "transmon0.measure.pulse_amp(0.25)\n", "device.add_element(transmon0)" ] }, { "cell_type": "markdown", "id": "12550ed8", "metadata": {}, "source": [ "The amplitude of the read-out pulse is now `0.25`, double what it was." ] }, { "cell_type": "code", "execution_count": 43, "id": "b516466d", "metadata": { "tags": [ "remove-cell" ] }, "outputs": [], "source": [ "from qblox_instruments import DummyBinnedAcquisitionData, DummyScopeAcquisitionData\n", "\n", "dummy_slot_idx = 1\n", "cluster.delete_dummy_binned_acquisition_data(slot_idx=dummy_slot_idx, sequencer=0)\n", "cluster.delete_dummy_binned_acquisition_data(slot_idx=dummy_slot_idx, sequencer=1)\n", "dummy_data_0 = [\n", " DummyBinnedAcquisitionData(data=(32, 0), thres=0, avg_cnt=0),\n", "]\n", "cluster.set_dummy_binned_acquisition_data(slot_idx=dummy_slot_idx, sequencer=0, acq_index_name=\"2\", data=dummy_data_0)" ] }, { "cell_type": "markdown", "id": "3308d45d", "metadata": {}, "source": [ "Let's compile the schedule." ] }, { "cell_type": "code", "execution_count": 44, "id": "fa78ea66", "metadata": {}, "outputs": [], "source": [ "from quantify_scheduler.backends import SerialCompiler\n", "\n", "compiler = SerialCompiler(name=\"compiler\")\n", "compiled_schedule = compiler.compile(schedule=schedule, config=device.generate_compilation_config())" ] }, { "cell_type": "markdown", "id": "2b48af26", "metadata": {}, "source": [ "Let's run the schedule, and retrieve the acquisitions." ] }, { "cell_type": "code", "execution_count": 45, "id": "0a53466b", "metadata": {}, "outputs": [], "source": [ "instrument_coordinator.prepare(compiled_schedule)\n", "instrument_coordinator.start()\n", "instrument_coordinator.wait_done(timeout_sec=10)\n", "\n", "acquisition = instrument_coordinator.retrieve_acquisition()" ] }, { "cell_type": "code", "execution_count": 46, "id": "175e657b", "metadata": {}, "outputs": [ { "data": { "text/html": [ "
\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "\n", "
<xarray.Dataset>\n",
       "Dimensions:      (acq_index_2: 1)\n",
       "Coordinates:\n",
       "  * acq_index_2  (acq_index_2) int64 0\n",
       "Data variables:\n",
       "    2            (acq_index_2) complex128 (0.26666666666666666+0j)
" ], "text/plain": [ "\n", "Dimensions: (acq_index_2: 1)\n", "Coordinates:\n", " * acq_index_2 (acq_index_2) int64 0\n", "Data variables:\n", " 2 (acq_index_2) complex128 (0.26666666666666666+0j)" ] }, "execution_count": 46, "metadata": {}, "output_type": "execute_result" } ], "source": [ "acquisition" ] }, { "cell_type": "markdown", "id": "6e1302a5", "metadata": {}, "source": [ "As you can see, because the measurement read-out pulse is now double in amplitude, the measured (acquired) value is now also double compared to the previous case." ] } ], "metadata": { "file_format": "mystnb", "kernelspec": { "display_name": "python3", "name": "python3" }, "language_info": { "codemirror_mode": { "name": "ipython", "version": 3 }, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.9.18" }, "mystnb": { "execution_timeout": 120 }, "source_map": [ 8, 36, 43, 47, 58, 72, 90, 96, 101, 103, 119, 121, 131, 133, 137, 169, 194, 198, 203, 209, 217, 219, 225, 237, 252, 254, 258, 300, 304, 323, 327, 332, 338, 346, 348, 403, 417, 431, 435, 440, 444, 452, 454, 458, 460, 492, 494, 499, 526, 530, 545, 549, 554, 560, 568, 570, 582, 590, 595, 604, 620, 639, 652, 656, 661, 667, 675, 677, 685, 692, 695, 699, 712, 716, 721, 725, 733, 735 ] }, "nbformat": 4, "nbformat_minor": 5 }