Backends
Backends allow execution of Qadence abstract quantum circuits. They could be chosen from a variety of simulators, emulators and hardware
and can enable circuit differentiability. The primary way to interact and configure
a backend is via the high-level API QuantumModel
.
Not all backends are equivalent
Not all backends support the same set of operations, especially while executing analog blocks. Qadence will throw descriptive errors in such cases.
Execution backends
PyQTorch: An efficient, large-scale simulator designed for
quantum machine learning, seamlessly integrated with the popular PyTorch deep learning framework for automatic differentiability.
It also offers analog computing for time-(in)dependent pulses. See PyQTorchBackend
.
Pulser: A Python library for pulse-level/analog control of
neutral atom devices. Execution via QuTiP. See PulserBackend
.
More: Proprietary Qadence extensions provide more high-performance backends based on tensor networks or differentiation engines.
For more enquiries, please contact: info@pasqal.com
.
Differentiation backend
The DifferentiableBackend
class enables different differentiation modes
for the given backend. This can be chosen from two types:
- Automatic differentiation (AD): available for PyTorch based backends (PyQTorch).
- Parameter Shift Rules (PSR): available for all backends. See this section for more information on differentiability and PSR.
In practice, only a diff_mode
should be provided in the QuantumModel
. Please note that diff_mode
defaults to None
:
import sympy
import torch
from qadence import Parameter, RX, RZ, Z, CNOT, QuantumCircuit, QuantumModel, chain, BackendName, DiffMode
x = Parameter("x", trainable=False)
y = Parameter("y", trainable=False)
fm = chain(
RX(0, 3 * x),
RX(0, x),
RZ(1, sympy.exp(y)),
RX(0, 3.14),
RZ(1, "theta")
)
ansatz = CNOT(0, 1)
block = chain(fm, ansatz)
circuit = QuantumCircuit(2, block)
observable = Z(0)
# DiffMode.GPSR is available for any backend.
# DiffMode.AD is only available for natively differentiable backends.
model = QuantumModel(circuit, observable, backend=BackendName.PYQTORCH, diff_mode=DiffMode.GPSR)
# Get some values for the feature parameters.
values = {"x": (x := torch.tensor([0.5], requires_grad=True)), "y": torch.tensor([0.1])}
# Compute expectation.
exp = model.expectation(values)
# Differentiate the expectation wrt x.
dexp_dx = torch.autograd.grad(exp, x, torch.ones_like(exp))
Low-level backend_factory
interface
Every backend in Qadence inherits from the abstract Backend
class:
Backend
and implement the following methods:
run
: propagate the initial state according to the quantum circuit and return the final wavefunction object.sample
: sample from a circuit.expectation
: computes the expectation of a circuit given an observable.convert
: convert the abstractQuantumCircuit
object to its backend-native representation including a backend specific parameter embedding function.
Backends are purely functional objects which take as input the values for the circuit parameters and return the desired output from a call to a method. In order to use a backend directly, embedded parameters must be supplied as they are returned by the backend specific embedding function.
Here is a simple demonstration of the use of the PyQTorch backend to execute a circuit in non-differentiable mode:
from qadence import QuantumCircuit, FeatureParameter, RX, RZ, CNOT, hea, chain
# Construct a feature map.
x = FeatureParameter("x")
z = FeatureParameter("y")
fm = chain(RX(0, 3 * x), RZ(1, z), CNOT(0, 1))
# Construct a circuit with an hardware-efficient ansatz.
circuit = QuantumCircuit(3, fm, hea(3,1))
The abstract QuantumCircuit
can now be converted to its native representation via the PyQTorch
backend.
from qadence import backend_factory
# Use only PyQtorch in non-differentiable mode:
backend = backend_factory("pyqtorch")
# The `Converted` object
# (contains a `ConvertedCircuit` with the original and native representation)
conv = backend.convert(circuit)
conv.circuit.original = ChainBlock(0,1,2)
├── ChainBlock(0,1)
│ ├── RX(0) [params: ['3*x']]
│ ├── RZ(1) [params: ['y']]
│ └── CNOT(0, 1)
└── ChainBlock(0,1,2) [tag: HEA]
├── ChainBlock(0,1,2)
│ ├── KronBlock(0,1,2)
│ │ ├── RX(0) [params: ['theta_0']]
│ │ ├── RX(1) [params: ['theta_1']]
│ │ └── RX(2) [params: ['theta_2']]
│ ├── KronBlock(0,1,2)
│ │ ├── RY(0) [params: ['theta_3']]
│ │ ├── RY(1) [params: ['theta_4']]
│ │ └── RY(2) [params: ['theta_5']]
│ └── KronBlock(0,1,2)
│ ├── RX(0) [params: ['theta_6']]
│ ├── RX(1) [params: ['theta_7']]
│ └── RX(2) [params: ['theta_8']]
└── ChainBlock(0,1,2)
├── KronBlock(0,1)
│ └── CNOT(0, 1)
└── KronBlock(1,2)
└── CNOT(1, 2)
conv.circuit.native = QuantumCircuit(
(operations): ModuleList(
(0): Sequence(
(operations): ModuleList(
(0): Sequence(
(operations): ModuleList(
(0): RX(target: (0,), param: 9af42b89-bde3-4667-95b8-d48aa811d0f9)
(1): RZ(target: (1,), param: b999f44c-e0ba-4efb-88f9-9cfcddf7547d)
(2): CNOT(control: (0,), target: (1,))
)
)
(1): Sequence(
(operations): ModuleList(
(0): Sequence(
(operations): ModuleList(
(0): Merge(
(operations): ModuleList(
(0): RX(target: (0,), param: 07ac2c73-83c6-488a-9582-c1e0f8f320c9)
(1): RY(target: (0,), param: e961cd3d-dd6a-4bee-a4d5-73b3e5986406)
(2): RX(target: (0,), param: 3f65ed7f-f54c-4207-a53c-2445f588c5b5)
)
)
(1): Merge(
(operations): ModuleList(
(0): RX(target: (1,), param: 36bc4c57-1d35-421c-8b71-39319ad0c374)
(1): RY(target: (1,), param: a56f2451-a0c2-4008-8d10-ace90618fab2)
(2): RX(target: (1,), param: f38f0959-2cd6-4b30-91d7-13992356a759)
)
)
(2): Merge(
(operations): ModuleList(
(0): RX(target: (2,), param: 8de573e3-d7f1-479d-a59f-4eb347b1d26a)
(1): RY(target: (2,), param: 9c9b858a-f01c-4a4c-9e57-393591e8b258)
(2): RX(target: (2,), param: f3327c9d-ae05-4d4e-8b2d-6b9631523b3d)
)
)
)
)
(1): Sequence(
(operations): ModuleList(
(0): Sequence(
(operations): ModuleList(
(0): CNOT(control: (0,), target: (1,))
)
)
(1): Sequence(
(operations): ModuleList(
(0): CNOT(control: (1,), target: (2,))
)
)
)
)
)
)
)
)
)
)
Additionally, Converted
contains all fixed and variational parameters, as well as an embedding
function which accepts feature parameters to construct a dictionary of circuit native parameters.
These are needed as each backend uses a different representation of the circuit parameters:
import torch
# Contains fixed parameters and variational (from the HEA)
conv.params
inputs = {"x": torch.tensor([1., 1.]), "y":torch.tensor([2., 2.])}
# get all circuit parameters (including feature params)
embedded = conv.embedding_fn(conv.params, inputs)
conv.params = {
theta_1: tensor([0.7195], requires_grad=True)
theta_6: tensor([0.2487], requires_grad=True)
theta_0: tensor([0.5327], requires_grad=True)
theta_5: tensor([0.8272], requires_grad=True)
theta_2: tensor([0.2177], requires_grad=True)
theta_7: tensor([0.3097], requires_grad=True)
theta_4: tensor([0.1798], requires_grad=True)
theta_3: tensor([0.4164], requires_grad=True)
theta_8: tensor([0.1463], requires_grad=True)
}
embedded = {
9af42b89-bde3-4667-95b8-d48aa811d0f9: tensor([3., 3.], grad_fn=<ViewBackward0>)
b999f44c-e0ba-4efb-88f9-9cfcddf7547d: tensor([2., 2.])
07ac2c73-83c6-488a-9582-c1e0f8f320c9: tensor([0.5327], grad_fn=<ViewBackward0>)
e961cd3d-dd6a-4bee-a4d5-73b3e5986406: tensor([0.4164], grad_fn=<ViewBackward0>)
3f65ed7f-f54c-4207-a53c-2445f588c5b5: tensor([0.2487], grad_fn=<ViewBackward0>)
36bc4c57-1d35-421c-8b71-39319ad0c374: tensor([0.7195], grad_fn=<ViewBackward0>)
a56f2451-a0c2-4008-8d10-ace90618fab2: tensor([0.1798], grad_fn=<ViewBackward0>)
f38f0959-2cd6-4b30-91d7-13992356a759: tensor([0.3097], grad_fn=<ViewBackward0>)
8de573e3-d7f1-479d-a59f-4eb347b1d26a: tensor([0.2177], grad_fn=<ViewBackward0>)
9c9b858a-f01c-4a4c-9e57-393591e8b258: tensor([0.8272], grad_fn=<ViewBackward0>)
f3327c9d-ae05-4d4e-8b2d-6b9631523b3d: tensor([0.1463], grad_fn=<ViewBackward0>)
}
With the embedded parameters, QuantumModel
methods are accessible:
output = tensor([[ 0.2192-0.0577j, 0.0861-0.0675j, 0.0410+0.1452j, -0.0411+0.3100j,
-0.6201-0.3545j, -0.3424-0.0385j, -0.0027+0.1880j, -0.1595+0.3555j],
[ 0.2192-0.0577j, 0.0861-0.0675j, 0.0410+0.1452j, -0.0411+0.3100j,
-0.6201-0.3545j, -0.3424-0.0385j, -0.0027+0.1880j, -0.1595+0.3555j]],
grad_fn=<TBackward0>)
Lower-level: the Backend
representation
If there is a requirement to work with a specific backend, it is possible to access directly the native circuit.
For example, should one wish to use PyQtorch noise features directly instead of using the NoiseHandler
interface from Qadence:
from pyqtorch.noise import Depolarizing
inputs = {"x": torch.rand(1), "y":torch.rand(1)}
embedded = conv.embedding_fn(conv.params, inputs)
# Define a noise channel on qubit 0
noise = Depolarizing(0, error_probability=0.1)
# Add noise to circuit
conv.circuit.native.operations.append(noise)
When running With noise, one can see that the output is a density matrix: