Skip to main content

A package for performing wire cuts of hardware without reset-gates or mid-circuit measurements. Built on top of qiskit

Project description

QCut

QCut is a quantum circuit knitting package for gate cuts and resrtless wire cuts. QCut has been designed to work with IQM's qpus, and therefore on the Finnish Quantum Computing Infrastructure (FiQCI), and tested with an IQM qpus. Additionally, QCut is built to be combatible with IQM’s Qiskit fork iqm_qiskit.

QCut has been built at CSC - IT Center for Science (Finnish IT Center for Science).

Check out jooniv.github.io/QCut/ for documentation and more examples.

Installation

For installation a UNIX-like system is currently needed due to pymetis being used for automatic cut finding. On Windows use WSL

Pip:
Installation should be done via uv

uv pip install QCut
#or
uv add QCut

Uv can be installed with

#Linux / mac
curl -LsSf https://astral.sh/uv/install.sh | sh

Note: for drawing circuits you might have to install pylatexenc. This can also be done with uv.

uv pip install pylatexenc
#or
uv add pylatexenc

Install from source
It is also possible to use QCut by cloning this repository and including it in your project folder.

cd QCut
uv pip install .
#or
uv sync --no-dev

#or with dev deps
uv sync

Usage

Manual Usage

1: Import needed packages

import QCut as ck
from QCut import cut, cutCZ
from qiskit import QuantumCircuit
from qiskit_aer import AerSimulator
from qiskit_aer.primitives import Estimator

2: Start by defining a QuantumCircuit just like in Qiskit

circuit  =  QuantumCircuit(3)
circuit.h(0)
circuit.cx(0,1)
circuit.cx(1,2)
   
circuit.measure_all()

circuit.draw("mpl")

3: Insert cut operations to the circuit to denote where we want to cut the circuit

Note that here we don't insert any measurements. Measurements will be automatically handled by QCut.

cut_circuit = QuantumCircuit(3)
cut_circuit.h(0)
cut_circuit.cx(0,1)
cut_circuit.append(cut, [1])
cut_circuit.cx(1,2)

cut_circuit.draw("mpl")

Or to use gate cuts one can do

cut_circuit = QuantumCircuit(3)
cut_circuit.h(0)
cut_circuit.h(1)
cut_circuit.append(cutCz, [0,1])
cut_circuit.h(1)
cut_circuit.cx(1,2)

cut_circuit.draw("mpl")

Note that currently QCut only supports cutting Cz gates so transformation have to be done manually for the time being (hence the added H gates)

4. Extract cut locations from cut_circuit and split it into independent subcircuit.

cut_locations, subcircuits, map_qubit = ck.get_locations_and_subcircuits(qc_cut)

Now we can draw our subcircuits.

subcircuits[0].draw("mpl")

subcircuits[1].draw("mpl")

5: Generate experiment circuits by inserting operations from a quasi-probability distribution for the identity channel

experiment_circuits, coefficients, id_meas = ck.get_experiment_circuits(subcircuits, cut_locations)

6: Run the experiment circuits

Here we are using the qisit AerSimulator as our backend but since QCut is backend independent you can choose whatever backend you want as long as you transpile the experiment circuits accordingly. QCut provides a function transpile_experiments() for doing just this.

Since QCut is a circuit knitting package the results are approximations of the actual values.

backend = AerSimulator()
results = ck.run_experiments(experiment_circuits, cut_locations, id_meas, backend=backend)

7. Define observables and calculate expectation values

Observables are Pauli-Z observables and are defined as a list of qubit indices. Multi-qubit observables are defined as a list inside the observable list.

If one wishes to calculate other than Pauli-Z observable expectation values currently this needs to be done by manually modifying the initial circuit to perform the basis transform.

observables = [0,1,2, [0,2]]
expectation_values = ck.estimate_expectation_values(results, coefficients, cut_locations, observables, map_qubit)

8: Finally calculate the exact expectation values and compare them to the results calculated with QCut

paulilist_observables = ck.get_pauli_list(observables, 3)

estimator = Estimator(run_options={"shots": None}, approximation=True)
exact_expvals = (
    estimator.run([circuit] * len(paulilist_observables),  # noqa: PD011
                  list(paulilist_observables)).result().values
)
import numpy as np

np.set_printoptions(formatter={"float": lambda x: f"{x:0.6f}"})

print(f"QCut expectation values:{np.array(expectation_values)}")
print(f"Exact expectation values with ideal simulator :{np.array(exact_expvals)}")

QCut circuit knitting expectation values: [0.007532 0.007532 -0.003662 1.010128]

Exact expectation values with ideal simulator :[0.000000 0.000000 0.000000 1.000000]

As we can see QCut is able to accurately reconstruct the expectation values. (Note that since this is a probabilistic method the results vary a bit each run)

Usage shorthand

For convenience, it is not necessary to go through each of the aforementioned steps individually. Instead, QCut provides a function run() that executes the whole wire-cutting sequence.

The same example can then be run like this:

backend = AerSimulator()
observables = [0,1,2, [0,2]]

estimated_expectation_values = ck.run(cut_circuit, observables, backend)

Automatic cuts

QCut comes with functionality for automatically finding good cut locations that can place both wire and gate cuts.

cut_locations, subcircuits, map_qubit = find_cuts(circuit , 3, cuts="both")
estimated_expectation_values = ck.run_cut_circuit(subcircuits, cut_locations, observables, map_qubit, backend)

Running on IQM fake backends

To use QCut with IQM's fake backends it is required to install Qiskit IQM. QCut supports version 17.8. Installation can be done with uv:

uv pip install qiskit-iqm==17.8
#or
uv add qiskit-iqm==17.8

After installation just import the backend you want to use:

from iqm.qiskit_iqm import IQMFakeAdonis()
backend = IQMFakeAdonis()

Running on FiQCI

For running on real hardware through the Lumi supercomputer's FiQCI partition follow the instructions here. If you are used to using Qiskit on jupyter notebooks it is recommended to use the Lumi web interface.

Running on other hardware

Running on other providers such as IBM is untested at the moment but as long as the hardware can be accessed with Qiskit QCut should be compatible.

Documentation

Check out jooniv.github.io/QCut/ for documentation and more examples.

The docs are built with sphinx using the sphinx book theme. To build the docs:

cd docs
uv pip install -r requirements-docs.txt
sphinx-build -v -b html . build/sphinx/html -W

HTML files can then be found under build/sphinx/html/

Acknowledgements

This project is built on top of Qiskit which is licensed under the Apache 2.0 license.

License

Apache 2.0 license

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

qcut-0.6.0.tar.gz (424.0 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

qcut-0.6.0-py3-none-any.whl (420.0 kB view details)

Uploaded Python 3

File details

Details for the file qcut-0.6.0.tar.gz.

File metadata

  • Download URL: qcut-0.6.0.tar.gz
  • Upload date:
  • Size: 424.0 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for qcut-0.6.0.tar.gz
Algorithm Hash digest
SHA256 ac94ccbb360d9e4c711a605f11d3ac9819e684e808e99837e4b5b7f46c425e4c
MD5 c683d4ddfb7020d36da3f5a39ba96e62
BLAKE2b-256 4102cbce18e8d997d005c03df76102adfa46de21e6c68ccdede5c2e12e8a7356

See more details on using hashes here.

Provenance

The following attestation bundles were made for qcut-0.6.0.tar.gz:

Publisher: publish_to_pypi.yml on JooNiv/QCut

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file qcut-0.6.0-py3-none-any.whl.

File metadata

  • Download URL: qcut-0.6.0-py3-none-any.whl
  • Upload date:
  • Size: 420.0 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for qcut-0.6.0-py3-none-any.whl
Algorithm Hash digest
SHA256 d7f427a2540176ed3f856e592de30e04d42e76b06e13a6da27da5ebdd8afc553
MD5 1bb835c27056bf9563ac570fee35e5e6
BLAKE2b-256 fb454f01ddfc92c9d2a8240d1eefe2a32a5b59484e51f74c66a0602112895db9

See more details on using hashes here.

Provenance

The following attestation bundles were made for qcut-0.6.0-py3-none-any.whl:

Publisher: publish_to_pypi.yml on JooNiv/QCut

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page