Source code for pennylane.labs.resource_estimation.ops.op_math.controlled_ops
# Copyright 2024 Xanadu Quantum Technologies Inc.# Licensed under the Apache License, Version 2.0 (the "License");# you may not use this file except in compliance with the License.# You may obtain a copy of the License at# http://www.apache.org/licenses/LICENSE-2.0# Unless required by applicable law or agreed to in writing, software# distributed under the License is distributed on an "AS IS" BASIS,# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.# See the License for the specific language governing permissions and# limitations under the License.r"""Resource operators for controlled operations."""fromtypingimportDictimportpennylaneasqmlimportpennylane.labs.resource_estimationasre# pylint: disable=arguments-differ,too-many-ancestors
[docs]classResourceCH(qml.CH,re.ResourceOperator):r"""Resource class for the CH gate. Resources: The resources are derived from the following identities (as presented in this `blog post <https://quantumcomputing.stackexchange.com/questions/15734/how-to-construct-a-controlled-hadamard-gate-using-single-qubit-gates-and-control>`_): .. math:: \begin{align} \hat{H} &= \hat{R}_{y}(\frac{\pi}{4}) \cdot \hat{Z} \cdot \hat{R}_{y}(\frac{-\pi}{4}), \\ \hat{Z} &= \hat{H} \cdot \hat{X} \cdot \hat{H}. \end{align} We can control on the Pauli-X gate to obtain our controlled Hadamard gate. """@staticmethoddef_resource_decomp(**kwargs)->Dict[re.CompressedResourceOp,int]:gate_types={}ry=re.ResourceRY.resource_rep()h=re.ResourceHadamard.resource_rep()cnot=re.ResourceCNOT.resource_rep()gate_types[h]=2gate_types[ry]=2gate_types[cnot]=1returngate_types
[docs]classResourceCY(qml.CY,re.ResourceOperator):r"""Resource class for the CY gate. Resources: The resources are derived from the following identity: .. math:: \hat{Y} = \hat{S} \cdot \hat{X} \cdot \hat{S}^{\dagger}. We can control on the Pauli-X gate to obtain our controlled-Y gate. """@staticmethoddef_resource_decomp(**kwargs)->Dict[re.CompressedResourceOp,int]:gate_types={}cnot=re.ResourceCNOT.resource_rep()s=re.ResourceS.resource_rep()gate_types[cnot]=1gate_types[s]=1+3# S^dagg = 3*S in cost TODO: Update with Adjoint(S)returngate_types
[docs]classResourceCZ(qml.CZ,re.ResourceOperator):r"""Resource class for the CZ gate. Resources: The resources are derived from the following identity: .. math:: \hat{Z} = \hat{H} \cdot \hat{X} \cdot \hat{H}. We can control on the Pauli-X gate to obtain our controlled-Z gate. """@staticmethoddef_resource_decomp(**kwargs)->Dict[re.CompressedResourceOp,int]:gate_types={}cnot=re.ResourceCNOT.resource_rep()h=re.ResourceHadamard.resource_rep()gate_types[cnot]=1gate_types[h]=2returngate_types
[docs]classResourceCSWAP(qml.CSWAP,re.ResourceOperator):r"""Resource class for the CSWAP gate. Resources: The resources are taken (figure 1d) from the paper `Shallow unitary decompositions of quantum Fredkin and Toffoli gates for connectivity-aware equivalent circuit averaging <https://arxiv.org/pdf/2305.18128>`_. The circuit which applies the SWAP operation on wires (1, 2) and controlled on wire (0) is given by: .. code-block:: bash 0: ────╭●────┤ 1: ─╭X─├●─╭X─┤ 2: ─╰●─╰X─╰●─┤ """@staticmethoddef_resource_decomp(**kwargs)->Dict[re.CompressedResourceOp,int]:gate_types={}tof=re.ResourceToffoli.resource_rep()cnot=re.ResourceCNOT.resource_rep()gate_types[tof]=1gate_types[cnot]=2returngate_types
[docs]classResourceCCZ(qml.CCZ,re.ResourceOperator):r"""Resource class for the CCZ gate. Resources: The resources are derived from the following identity: .. math:: \hat{Z} = \hat{H} \cdot \hat{X} \cdot \hat{H}. We replace the Pauli-X gate with a Toffoli gate to obtain our control-control-Z gate. """@staticmethoddef_resource_decomp(**kwargs)->Dict[re.CompressedResourceOp,int]:gate_types={}toffoli=re.ResourceToffoli.resource_rep()h=re.ResourceHadamard.resource_rep()gate_types[toffoli]=1gate_types[h]=2returngate_types
[docs]classResourceCNOT(qml.CNOT,re.ResourceOperator):r"""Resource class for the CNOT gate. Resources: There is no further decomposition provided for this gate. """@staticmethoddef_resource_decomp(**kwargs)->Dict[re.CompressedResourceOp,int]:raisere.ResourcesNotDefined
[docs]classResourceToffoli(qml.Toffoli,re.ResourceOperator):r"""Resource class for the Toffoli gate. Resources: The resources are obtained from (in figure 1.) the paper `Novel constructions for the fault-tolerant Toffoli gate <https://arxiv.org/pdf/1212.5069>`_. The circuit which applies the Toffoli gate on target wire 'target' with control wires ('c1', 'c2') is given by: .. code-block:: bash c1: ─╭●────╭X──T†────────╭X────╭●───────────────╭●─┤ c2: ─│──╭X─│──╭●───T†─╭●─│──╭X─│────────────────╰Z─┤ aux1: ─╰X─│──│──╰X───T──╰X─│──│──╰X────────────────║─┤ aux2: ──H─╰●─╰●──T─────────╰●─╰●──H──S─╭●──H──┤↗├──║─┤ target: ─────────────────────────────────╰X──────║───║─┤ ╚═══╝ """@staticmethoddef_resource_decomp(**kwargs)->Dict[re.CompressedResourceOp,int]:gate_types={}cnot=re.ResourceCNOT.resource_rep()t=re.ResourceT.resource_rep()h=re.ResourceHadamard.resource_rep()s=re.ResourceS.resource_rep()cz=re.ResourceCZ.resource_rep()gate_types[cnot]=9gate_types[h]=3gate_types[s]=1gate_types[cz]=1gate_types[t]=2+2*(7)# T^dagg = 7*T in cost TODO: Update with Adjoint(T)returngate_types
[docs]@staticmethoddeftextbook_resource_decomp()->Dict[re.CompressedResourceOp,int]:r"""Resources for the Toffoli gate Resources: The resources are taken (figure 4.9) from the textbook `Quantum Computation and Quantum Information <https://www.cambridge.org/highereducation/books/quantum-computation-and-quantum-information/01E10196D0A682A6AEFFEA52D53BE9AE#overview>`_. The circuit is given by: .. code-block:: bash 0: ───────────╭●───────────╭●────╭●──T──╭●─┤ 1: ────╭●─────│─────╭●─────│───T─╰X──T†─╰X─┤ 2: ──H─╰X──T†─╰X──T─╰X──T†─╰X──T──H────────┤ """gate_types={}cnot=re.ResourceCNOT.resource_rep()t=re.ResourceT.resource_rep()h=re.ResourceHadamard.resource_rep()gate_types[cnot]=6gate_types[h]=2gate_types[t]=7returngate_types
[docs]classResourceMultiControlledX(qml.MultiControlledX,re.ResourceOperator):r"""Resource class for the MultiControlledX gate. Resources: The resources are obtained from (table 3.) the paper `Polylogarithmic-depth controlled-NOT gates without ancilla qubits <https://www.nature.com/articles/s41467-024-50065-x>`_. Specifically, the resources are given by the following rules: * If there is only one control qubit, treat the resources as a :code:`CNOT` gate. * If there are two control qubits, treat the resources as a :code:`Toffoli` gate. * If there are three control qubits, the resources are two :code:`CNOT` gates and one :code:`Toffoli` gate. * If there are more than three control qubits (:math:`n`), the resources are given by :math:`36n - 111` :code:`CNOT` gates. """# TODO: There is a more efficient resource decomposition, need to update this based on the paper.@staticmethoddef_resource_decomp(num_ctrl_wires,num_ctrl_values,num_work_wires,**kwargs,# pylint: disable=unused-argument)->Dict[re.CompressedResourceOp,int]:gate_types={}ifnum_ctrl_values:x=re.ResourceX.resource_rep()gate_types[x]=num_ctrl_values*2cnot=re.ResourceCNOT.resource_rep()ifnum_ctrl_wires==1:gate_types[cnot]=1returngate_typestoffoli=re.ResourceToffoli.resource_rep()ifnum_ctrl_wires==2:gate_types[toffoli]=1returngate_typesifnum_ctrl_wires==3:gate_types[cnot]=2gate_types[toffoli]=1returngate_typesgate_types[cnot]=36*num_ctrl_wires-111returngate_types
[docs]classResourceCRX(qml.CRX,re.ResourceOperator):r"""Resource class for the CRX gate. Resources: The resources are taken from (in figure 1b.) the paper `T-count and T-depth of any multi-qubit unitary <https://arxiv.org/pdf/2110.10292>`_. In combination with the following identity: .. math:: \hat{RX} = \hat{H} \cdot \hat{RZ} \cdot \hat{H}, we can express the :code:`CRX` gate as a :code:`CRZ` gate conjugated by :code:`Hadamard` gates. The expression for controlled-RZ gates is used as defined in the reference above. """@staticmethoddef_resource_decomp(**kwargs)->Dict[re.CompressedResourceOp,int]:gate_types={}h=re.ResourceHadamard.resource_rep()rz=re.ResourceRZ.resource_rep()cnot=re.ResourceCNOT.resource_rep()gate_types[cnot]=2gate_types[rz]=2gate_types[h]=2returngate_types
[docs]classResourceCRY(qml.CRY,re.ResourceOperator):r"""Resource class for the CRY gate. Resources: The resources are taken from (in figure 1b.) the paper `T-count and T-depth of any multi-qubit unitary <https://arxiv.org/pdf/2110.10292>`_. The resources are derived with the following identity: .. math:: \hat{RY}(\theta) = \hat{X} \cdot \hat{RY}(- \theta) \cdot \hat{X}. By replacing the :code:`X` gates with :code:`CNOT` gates, we obtain a controlled-version of this identity. Thus we are able to constructively or destructively interfere the gates based on the value of the control qubit. """@staticmethoddef_resource_decomp(**kwargs)->Dict[re.CompressedResourceOp,int]:gate_types={}cnot=re.ResourceCNOT.resource_rep()ry=re.ResourceRY.resource_rep()gate_types[cnot]=2gate_types[ry]=2returngate_types
[docs]classResourceCRZ(qml.CRZ,re.ResourceOperator):r"""Resource class for the CRZ gate. Resources: The resources are obtained from (in figure 1b.) the paper `T-count and T-depth of any multi-qubit unitary <https://arxiv.org/pdf/2110.10292>`_. They are derived from the following identity: .. math:: \hat{RZ}(\theta) = \hat{X} \cdot \hat{RZ}(- \theta) \cdot \hat{X}. By replacing the :code:`X` gates with :code:`CNOT` gates, we obtain a controlled-version of this identity. Thus we are able to constructively or destructively interfere the gates based on the value of the control qubit. """@staticmethoddef_resource_decomp(**kwargs)->Dict[re.CompressedResourceOp,int]:gate_types={}cnot=re.ResourceCNOT.resource_rep()rz=re.ResourceRZ.resource_rep()gate_types[cnot]=2gate_types[rz]=2returngate_types
[docs]classResourceCRot(qml.CRot,re.ResourceOperator):r"""Resource class for the CRot gate. Resources: The resources are derived from (in figure 1b.) the paper `T-count and T-depth of any multi-qubit unitary <https://arxiv.org/pdf/2110.10292>`_. The resources are derived with the following identities: .. math:: \begin{align} \hat{RZ}(\theta) = \hat{X} \cdot \hat{RZ}(- \theta) \cdot \hat{X}, \\ \hat{RY}(\theta) = \hat{X} \cdot \hat{RY}(- \theta) \cdot \hat{X}. \end{align} This identity is applied along with some clever choices for the angle values to combine rotation; the final circuit takes the form: .. code-block:: bash ctrl: ─────╭●─────────╭●─────────┤ trgt: ──RZ─╰X──RZ──RY─╰X──RY──RZ─┤ """@staticmethoddef_resource_decomp(**kwargs)->Dict[re.CompressedResourceOp,int]:gate_types={}cnot=re.ResourceCNOT.resource_rep()rz=re.ResourceRZ.resource_rep()ry=re.ResourceRY.resource_rep()gate_types[cnot]=2gate_types[rz]=3gate_types[ry]=2returngate_types
[docs]classResourceControlledPhaseShift(qml.ControlledPhaseShift,re.ResourceOperator):r"""Resource class for the ControlledPhaseShift gate. Resources: The resources are derived using the fact that a :code:`PhaseShift` gate is identical to the :code:`RZ` gate up to some global phase. Furthermore, a controlled global phase simplifies to a :code:`PhaseShift` gate. This gives rise to the following identity: .. math:: CR_\phi(\phi) = (R_\phi(\phi/2) \otimes I) \cdot CNOT \cdot (I \otimes R_\phi(-\phi/2)) \cdot CNOT \cdot (I \otimes R_\phi(\phi/2)) """@staticmethoddef_resource_decomp(**kwargs)->Dict[re.CompressedResourceOp,int]:gate_types={}cnot=re.ResourceCNOT.resource_rep()rz=re.ResourceRZ.resource_rep()gate_types[cnot]=2gate_types[rz]=3returngate_types