Library to connect to Quantum Interface's QiController.
Project description
qiclib Python package
qiclib is Quantum Interface's official Python client for the QiController.
The QiController is currently developed at the Institute for Data Processing and Electronics from Karlsruhe Institute of Technology (KIT).
Installing
To install the qiclib and make it available system-wide using, use:
pip install qiclib
or
uv pip install qiclib
Developing
Important: This section describes development using the internal, Gitlab based repository, not the public GitHub based mirror.
The recommended development process involves uv. Install the tool as mentioned in the documentation.
To install, clone this repository and initialize it using uv;
git clone git@gitlab.kit.edu:kit/ipe-sdr/ipe-sdr-dev/software/qiclib.git --recurse-submodules
cd qiclib
uv sync
You can then refer to the Testing section to verify that the setup is correct.
The usage of pre-commit is recommended and you can install it using
uvx pre-commit install
Visual Studio Code
This repository includes settings and recommended extensions for VSCode. If you open this repository using VSCode for the first time, a popup will appear on the bottom-right corner asking for permission to install recommended extensions.
If you want to inspect these recommendations or have accidentally discarded the notification, go to the Notifications tab and search for @recommended.
Using jupyter
If you have installed qiclib system-wide, or in a virtual environment, you can import qiclib simply by calling import qiclib.
Example notebooks can be found in the examples subdirectory.
Development with jupyter
By default, uv sync will also install jupyter so no further setup is required.
It is recommended to store notebooks in the Notebooks folder that are used for development and testing purposes exclusively.
This folder is ignored in git, but uses the same virtual environment as qiclib.
Testing
Run
uv run pytest
to run all unittests
Getting Started
The QiController needs to be powered and connected to the same network as the control computer. It can then be accessed via its IP address or host name.
Everything is based on the QiController driver class, so simply import qiclib and establish a connection:
import qiclib as ql
from qiclib.code import *
qic = ql.QiController('IP ADDRESS OR HOST NAME')
qiclib includes a high-level description language called QiCode (already imported using the second line above). With it, you can conveniently specify your experiments in an easy and intuitive manner.
QiCode lets you specify experiments in a generic way, so-called QiJobs, and then execute them on your superconducting qubit chip.
The physical properties of your sample are stored inside a QiSample object.
The sample can consist of one or more cells.
Each cell corresponds to a qubit and defines all relevant properties for this qubit.
This can be pulse lengths, frequencies, but also other experiment-related parameters.
For this introduction, we will stick with one qubit and thus one cell:
sample = QiSample(1) # 1 cell/qubit only
sample[0]["rec_pulse"] = 416e-9 # s readout pulse length
sample[0]["rec_length"] = 400e-9 # s recording window size
sample[0]["rec_frequency"] = 60e6 # Hz readout pulse frequency
sample[0]["manip_frequency"] = 80e6 # Hz control pulse frequency
One can define as many properties as one likes. The naming convention of these properties is left completely up to you. However, if you want to use pre-built experiments from qiclib, you should use the same property names as are used there. You can also store qubit-related characteristics, like decay times and further information which are not needed for the experiments. As the sample can be exported as JSON file and imported again later, this can become quite useful.
In the sample, we already provided some information on how our readout pulse should look like and how long our recording window should be.
At the beginning, one now typically wants to calibrate the electrical delay of the readout pulse through the experimental setup.
qiclib offers an automated scheme for this purpose, optimizing for the highest signal amplitude at the IF frequency.
Of course, manual calibration (see other methods inside ql.init) is also possible.
ql.init.calibrate_readout(qic, sample, averages=1000)
This will perform multiple experiments on the QiController to determine the electrical delay and to record the final readout window.
It will also plot you the resulting data using matplotlib.
The optimal delay will be stored as "rec_offset" inside the sample object so it can be used for the following experiments.
Congratulation! You have successfully put the QiController into operation. Now you can continue reading how to use QiCode for your custom experiments.
QiCode Usage
The basic commands and classes that are provided for building your experiments with QiCode are:
-
QiPulse(length, shape, amplitude, phase)Creates a pulse object that can be used in other commands
-
Play(cell, pulse)Play the given
pulseat the manipulation output for the givencell -
PlayReadout(cell, pulse)Same as
Playbut for the readout pulse output -
Recording(cell, duration, offset, save_to, state_to)Performs a recording at the input of the cell of given
durationandoffset(electrical delay). Typically used directly after aPlayReadoutcommand. Withsave_to, the result data can be stored and labeled by the given string. Withstate_to, the obtained state can be saved to a QiVariable. -
Wait(cell, delay)Waits the given
delayin the execution time of thecellbefore continuing with the next command. -
QiVariable(type),QiTimeVariable()andQiStateVariable()Creates a variable that can be used during the control flow or to temporarily store a measured qubit state.
Furthermore, there are context managers (which are used with the with statement) to represent control logic:
-
with If(condition):Conditional branching, only executes the indented block that follows if the condition is true.
-
with Else():Can follow after
with Ifand does exactly what you would expect it to do. -
with ForRange(variable, start, end, step):The passed
variablewill be looped fromstarttoend(excluded) withstepincrements. The following idented block will be repeated for each value ofvariable.
One nice thing about QiCode is that you can define reusable building blocks for your experiments. We call these QiGates and annotate them accordingly (@QiGate). Together with the property names of the QiSample, one can really reuse them for different qubits without any adaptations to the gates:
@QiGate
def Readout(cell: QiCell, save_to: str = None):
PlayReadout(cell, QiPulse(cell["rec_pulse"], frequency=cell["rec_frequency"]))
Recording(
cell,
duration=cell["rec_length"],
offset=cell["rec_offset"],
save_to=save_to
)
@QiGate
def PiPulse(cell: QiCell):
Play(cell, QiPulse(cell["pi"], frequency=cell["manip_frequency"]))
@QiGate
def Thermalize(cell: QiCell):
Wait(cell, 5 * cell["T1"])
As with the basic commands, the QiGates typically start by defining the cell on which they act. This is not strictly required though. Furthermore, also multi-qubit gates can be implemented in the same manner acting on multiple cells.
Now, one can define a first experiment. In QiCode, experiments are called QiJob and written in an abstract way so they can be easily reused for different samples. Let us consider a Rabi experiment:
# Commands are always encapsulated within the QiJob context
with QiJob() as rabi:
# First, we define how many qubits the experiment requires
q = QiCells(1)
# Rabi consists of variable length excitation pulses,
# so we need to create a time variable
length = QiTimeVariable()
# The variable can then be changed within a for loop
with ForRange(length, 0, 1e-6, 20e-9):
# Output the manipulation pulse with variable length
Play(q[0], QiPulse(length, frequency=q[0]["manip_frequency"]))
# Perform a consecutive readout (using the above QiGate)
# The data can later by accessed via the specified name "result"
Readout(q[0], save_to="result")
# Wait for the qubit to thermalize (also a QiGate)
Thermalize(q[0])
The QiCells object inside the QiJob only acts as placeholder for the real qubit sample and will be replaced by it once we run the experiment.
The placeholder specifies how many cells/qubits the experiment is written for, so one in this case.
All cell properties that are needed for the Rabi experiment are already defined in our sample above where we just calibrated the readout.
So we can just run the job by passing both QiController and QiSample object to it:
rabi.run(qic, sample, averages=1000)
data = rabi.cells[0].data("result")
This will execute the whole job including the for loop and repeat and average the whole experiment 1000 times. The resulting data object will look like this:
[
# Averaged I results (len = 50 in this example)
np.array([...]),
# Averaged Q results (same len)
np.array([...])
]
Job description for the most common experiments are readily available as ql.jobs.* functions, like ql.jobs.Rabi or ql.jobs.T1.
More information on QiCode can be found here.
Qkit Integration
This package is best used together with the quantum measurement suite Qkit which has to be installed separately. Stand-alone usage of qiclib with reduced capabilities is also possible.
After installation, convenience features like storing sample objects (see qkit.measure.samples_class) and displaying progress bars within Jupyter(lab) notebooks are available.
The following code demonstrates how both packages can be easily combined to perform a Rabi experiment (based on the code above):
from qkit.measure.timedomain.measure_td import Measure_td
# We use the rabi QiJob from above and extract an experiment object from it
exp = rabi.create_experiment(qic, sample, averages=1000)
# Create Qkit measurement object and pass an adapter object as Qkit sample
m = Measure_td(exp.qkit_sample)
m.set_x_parameters(exp.time_range(0, 1e-6, 20e-9), 'pulse_length', None, 's')
m.dirname = 'rabi'
# Start the measurement, and lean back:
m.measure_1D_AWG(iterations=1)
Everything else is handled for you by Qkit and qiclib! Have fun performing your experiments.
Contributing
Ideas for new features and bug reports are highly welcome. Please use the issue tracker for this purpose.
LICENSE
qiclib is released under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or any later version.
Please see the COPYING files for details.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file qiclib-2.0.0.tar.gz.
File metadata
- Download URL: qiclib-2.0.0.tar.gz
- Upload date:
- Size: 5.0 MB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.6.9
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
d504b263c7561fc8bc68bc28c105e0049d9929ca930db283de1cb82760fc85b5
|
|
| MD5 |
bcc7ddc9cd8eeeaea41a9b121042c649
|
|
| BLAKE2b-256 |
0574f77ae4d40f1ea9135679f01f452b81ac1a6b7e7b3d1db82a09383055d55e
|
File details
Details for the file qiclib-2.0.0-py3-none-any.whl.
File metadata
- Download URL: qiclib-2.0.0-py3-none-any.whl
- Upload date:
- Size: 678.3 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.6.9
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
0a3537cde5cfb5f9e9344e00302b025adc4ef501de185e0e2ff1f4c9a3a2be02
|
|
| MD5 |
780b716e75cc7db3ab6be28e4ebae01a
|
|
| BLAKE2b-256 |
f1e03a5275fb4d7ee497313ebf4ed3c7a06f2fe7151d7209962425bfed1505d5
|