GHZ State preparation and noise¶
In this example, we will illustrate how to work with bloqade
's heuristic noise models of Gemini class digital quantum processors by applying them to a circuit that prepares a GHZ state.
Primer on Gemini noise models¶
In bloqade
, there are two classes of heuristic noise models: one-zone models such as GeminiOneZoneNoiseModel
and a two-zone model GeminiTwoZoneNoiseModel
.
These are inspired by two distinct approaches to implement a quantum circuit on hardware and are designed to get a sense of the influence of noise on Gemini class hardware.
On the one hand, the one-zone model assumes a single-zone layout where qubits remain in the gate zone throughout the computation. On the other hand, the two-zone model incorporates a storage zone and assumes that qubits are transported between gate and storage regions.
Both models are informed by benchmark data on the device but are intentionally conservative. Specifically, they tend to overestimate noise due to the lack of knowledge about optimized move schedules, which leads to overestimating move-induced errors.
At this stage, we recommend interpreting the two models as providing a range for expected noise levels on Gemini-class devices, rather than precise predictions. They are useful for gaining intuition about noise sensitivity and for benchmarking algorithmic robustness to errors, using hardware-informed but simplistic assumptions.
Note, that there are actually two additional one-zone noise models, GeminiOneZoneNoiseModelCorrelated
and GeminiOneZoneNoiseModelConflictGraphMoves
.
As the names suggest, the former also takes into account correlated noise, whereas the latter takes into account more realistic move schedules.
In the following example, we will not be considering these two, but they are interchangeable with the used noise models (up to the fact, that the conflict graph moves require you to specify qubits as cirq.GridQubit
s).
Noise model implementations¶
For now, these noise models are implemented as cirq.NoiseModel
classes, so that you can use with any circuit you build using cirq
.
They are part of the bloqade.cirq_utils
submodule.
Support for using these models with e.g. squin will follow in the future. However, you can already rely on interoperability with cirq in order to convert (noisy) circuits to squin kernels and use other parts of the compiler pipeline.
GHZ preparation and noise¶
Now, let's get started with the actual example.
As a first step, we will define a function that builds a GHZ circuit in cirq that has a depth linear in the number of qubits.
import cirq
import numpy as np
import matplotlib.pyplot as plt
from bloqade.cirq_utils import noise, transpile
from bloqade import squin
def ghz_circuit(n: int) -> cirq.Circuit:
qubits = cirq.LineQubit.range(n)
# Step 1: Hadamard on the first qubit
circuit = cirq.Circuit(cirq.H(qubits[0]))
# Step 2: CNOT chain from qubit i to i+1
for i in range(n - 1):
circuit.append(cirq.CNOT(qubits[i], qubits[i + 1]))
return circuit
Closer look at a basic circuit¶
Here's what this circuit looks like for n=3
qubits:
ghz_circuit_3 = ghz_circuit(3)
print(ghz_circuit_3)
So far so good.
Now, we will convert the circuit above to a noisy one using bloqade's cirq_utils
submodule.
Specifically, we can use the noise.transform_circuit
utility function with a noise model of our choice.
noise_model = noise.GeminiOneZoneNoiseModel()
noisy_ghz_circuit_3 = noise.transform_circuit(ghz_circuit_3, model=noise_model)
print(noisy_ghz_circuit_3)
As you can see, we have successfully added noise. However, the circuit also looks very different in terms of its gates.
This is because noise.transform_circuit
does actually two things:
- Since we want to consider a circuit that is compatible with the Gemini architecture, we need to transform it to the native gate set first. This set consists of (phased) X gates and CZ gates only.
- Once we have a native circuit, noise is injected according to the chosen noise model.
To clarify, here is how you would convert the circuit without using the noise.transform_circuit
utility function:
native_ghz_3 = transpile(ghz_circuit_3)
print(native_ghz_3)
Note that transpile
basically just wraps cirq's own cirq.optimize_for_target_gateset(circuit, gateset=cirq.CZTargetGateset())
, with some additional benefits (such as filtering out empty moments).
Using this native circuit, we can obtain the same noisy circuit as before by simply using cirq's cirq.Circuit.with_noise
method.
noisy_ghz_circuit_3 = native_ghz_3.with_noise(noise_model)
print(noisy_ghz_circuit_3)
Studying the fidelity¶
Now that we have got the basics down, we can compute the fidelity of noisy circuits with different qubit numbers. By fidelity, we simply mean the overlap of the final state with the perfect GHZ state expected from the noise-less version of the circuit.
The corresponding density matrices are obtained using cirq
's simulator.
We will do the simulation using two different noise models, the one-zone model used above and also the two-zone model.
Fidelity calculation
In the following, we will simply use the expectation value of the noisy density matrix computed against the noiseless one as a proxy for fidelity. This is a suboptimal choice, but we wanted to keep the example simple. Feel free to substitute the fidelity calculation by the fidelity of your choice (e.g. the Uhlmann fidelity)
qubits = range(3, 9)
one_zone_model = noise.GeminiOneZoneNoiseModel()
two_zone_model = noise.GeminiTwoZoneNoiseModel()
simulator = cirq.DensityMatrixSimulator()
fidelities_one_zone = []
fidelities_two_zone = []
for n in qubits:
circuit = ghz_circuit(n)
one_zone_circuit = noise.transform_circuit(circuit, model=one_zone_model)
two_zone_circuit = noise.transform_circuit(circuit, model=two_zone_model)
rho = simulator.simulate(circuit).final_density_matrix
rho_one_zone = simulator.simulate(one_zone_circuit).final_density_matrix
rho_two_zone = simulator.simulate(two_zone_circuit).final_density_matrix
fidelity_one_zone = np.trace(rho @ rho_one_zone)
fidelity_two_zone = np.trace(rho @ rho_two_zone)
fidelities_one_zone.append(fidelity_one_zone)
fidelities_two_zone.append(fidelity_two_zone)
Now, let's have a look at the results.
plt.plot(qubits, fidelities_one_zone, "o", label="one-zone model")
plt.plot(qubits, fidelities_two_zone, "x", label="two-zone model")
plt.xlabel("Number of qubits")
plt.ylabel("Fidelity")
plt.legend()
We can see that in both cases the fidelity goes down when increasing the number of qubits.
Interestingly, there is a cross-over point where the two-zone model starts to exhibit a better fidelity. This is because as the number of qubits grows, the error introduced on idle qubits inside the gate zone is larger in the one-zone model since all qubits are always inside the gate zone. Whereas, in the two-zone model, qubits are moved between the gate and storage zones.
You could now think about how to optimize the circuits in order to reduce their sensitivity for noise. For example, you can reduce the circuit depth
Modifying the noise¶
There are a number of parameters that govern the effect a noise model introduces into a circuit. These can all be set independently to adapt the noise model to your specific application.
In general, there are noise parameters for the following noise processes:
- Depolarization due to gate application.
- Depolarization due to movement, both applied to moving atoms and idle atoms (a.k.a. sitter errors).
- Atom loss errors.
Atom loss
Please note, that atom loss is currently not supported, i.e. it's not considered in the noise models. We plan to add that in the future.
The noise processes are further split into local and global noise channels and separated by their cause.
For a full list of noise parameters and a description of each one, please refer to the move noise model in bloqade.qasm2.dialects.noise.model.MoveNoiseModelABC
We can use those parameters in order to modify the strength of the noise.
For example, say you want to introduce an extra penalty for moving qubits around in order to study how you can reduce movements. To do so, let's re-use the fidelity calculation using the two-zone model from above, but modify movement errors.
We can query the default move errors from the cirq
noise model:
default_model = noise.GeminiTwoZoneNoiseModel()
px, py, pz = default_model.mover_px, default_model.mover_py, default_model.mover_pz
print(
f"The noise Pauli channel associated with moving atoms is (px, py, pz) = ({px,py,pz})."
)
Then we can instantiate a noise model with modified parameters:
modified_two_zone_model = noise.GeminiTwoZoneNoiseModel(
mover_px=2e-3,
mover_py=2e-3,
mover_pz=3e-3,
)
fidelities_modified_two_zone = []
for n in qubits:
circuit = ghz_circuit(n)
noisy_circuit = noise.transform_circuit(circuit, model=modified_two_zone_model)
rho = simulator.simulate(circuit).final_density_matrix
rho_noisy = simulator.simulate(noisy_circuit).final_density_matrix
fidelities_modified_two_zone.append(np.trace(rho @ rho_noisy))
plt.plot(qubits, fidelities_one_zone, "o", label="one-zone model")
plt.plot(qubits, fidelities_modified_two_zone, "x", label="modified two-zone model")
plt.xlabel("Number of qubits")
plt.ylabel("Fidelity")
plt.legend()
As you can see, the fidelities no longer cross over since the increased movement noise now eliminates the advantage of the two-zone model for the considered numbers of qubits.
Interoperability with squin¶
Finally, we want to point out that you can also use the generated noisy circuits to obtain a squin kernel function.
This is useful if you want to use other features of the bloqade pipeline.
For example, it would allow you to run the pyqrack
simulator instead of cirq
's own, which can be more efficient.
circuit = ghz_circuit(5)
noisy_circuit = noise.transform_circuit(circuit, model=noise.GeminiOneZoneNoiseModel())
kernel = squin.cirq.load_circuit(circuit, kernel_name="kernel")
noisy_kernel = squin.cirq.load_circuit(noisy_circuit, kernel_name="noisy_kernel")
kernel.print()