Source code for qrisp.environments.GMS_environment
"""
\********************************************************************************
* Copyright (c) 2023 the Qrisp authors
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0.
*
* This Source Code may also be made available under the following Secondary
* Licenses when the conditions for such availability set forth in the Eclipse
* Public License, v. 2.0 are satisfied: GNU General Public License, version 2
* with the GNU Classpath Exception which is
* available at https://www.gnu.org/software/classpath/license.html.
*
* SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
********************************************************************************/
"""
from qrisp.environments.quantum_environments import QuantumEnvironment
from qrisp.misc.GMS_tools import GXX_converter
# Environments that allows quick and easy access to the GMS_converter
# To use it make sure every gate that happens inside is either phase or cphase
[docs]
class GMSEnvironment(QuantumEnvironment):
"""
This environment provides a convenient interface for constructing quantum algorithms
using the Ion-trap native GMS gates. GMS gates allow entangling more than 2 qubits
in a single step and can therefore boost performance in many situations. For more
information on GMS gates consult https://arxiv.org/abs/quant-ph/9810040 . The
techniques for converting the circuits presented to this environment are mostly
based on https://ieeexplore.ieee.org/document/9815035 .
This environment allows to code blocks of phase-only gates as we are used to but
compiles these blocks to GMS gates.
Examples
--------
We create a function performing the quantum Fourier-transform using GMS gates: ::
from qrisp import QuantumEnvironment, GMSEnvironment, h, cp, swap
import numpy as np
def QFT(qv, use_gms = False):
n = qv.size
if use_gms:
env = GMSEnvironment
else:
env = QuantumEnvironment
for i in range(n):
h(qv[i])
if i == n-1:
break
with env():
#This is the block which converted to GMS gates
#We can only use the gates p, cp and rz in here
for k in range(n-i-1):
cp(2*np.pi/2**(k+2), qv[k+i+1], qv[i])
for i in range(n//2):
swap(qv[i], qv[n-i-1])
We inspect the resulting quantum circuit:
>>> from qrisp import QuantumFloat, invert
>>> qf = QuantumFloat(5)
>>> qf[:] = 13
>>> QFT(qf, use_gms = True)
>>> print(qf.qs)
::
QuantumCircuit:
--------------
┌───┐┌───┐┌─────────────────────┐ »
qf.0: ┤ X ├┤ H ├┤0 ├─────────────────────────────────»
└───┘└───┘│ │┌───┐┌─────────────────────┐ »
qf.1: ──────────┤1 ├┤ H ├┤0 ├─────»
┌───┐ │ │└───┘│ │┌───┐»
qf.2: ┤ X ├─────┤2 GXX converted gate ├─────┤1 ├┤ H ├»
├───┤ │ │ │ GXX converted gate │└───┘»
qf.3: ┤ X ├─────┤3 ├─────┤2 ├─────»
└───┘ │ │ │ │ »
qf.4: ──────────┤4 ├─────┤3 ├─────»
└─────────────────────┘ └─────────────────────┘ »
«
«qf.0: ─────────────────────────────────────────────────────────X─
« │
«qf.1: ─────────────────────────────────────────────────────X───┼─
« ┌─────────────────────┐ │ │
«qf.2: ┤0 ├──────────────────────────────┼───┼─
« │ │┌───┐┌─────────────────────┐ │ │
«qf.3: ┤1 GXX converted gate ├┤ H ├┤0 ├──X───┼─
« │ │└───┘│ GXX converted gate │┌───┐ │
«qf.4: ┤2 ├─────┤1 ├┤ H ├─X─
« └─────────────────────┘ └─────────────────────┘└───┘
Live QuantumVariables:
---------------------
QuantumFloat qf
Now we check that the GMS version indeed performs the same operation as the CNOT
version by performing the inverse of the CNOT version.
>>> with invert(): QFT(qf, use_gms = False)
>>> print(qf)
{13: 1.0}
"""
# We only need to modify the compile method of the base environment class
def compile(self):
# Temporarily store the data of the quantum session
temp_data = list(self.env_qs.data)
self.env_qs.clear_data()
QuantumEnvironment.compile(self)
copied_circ = self.env_qs.copy()
depth_dic = self.env_qs.get_depth_dic()
i = 0
while i < len(copied_circ.qubits):
if depth_dic[copied_circ.qubits[i]]:
i += 1
else:
copied_circ.qubits.pop(i)
if not len(copied_circ.qubits):
return
# Convert the circuit
# converted_gate, qubit_map = GMS_converter(self.env_qs, True)
converted_gate = GXX_converter(copied_circ).to_gate()
converted_gate.name = "GXX converted gate"
self.env_qs.clear_data()
# Recover original circuit
self.env_qs.data.extend(temp_data)
# Apply original circuit
self.env_qs.append(converted_gate, copied_circ.qubits)