Source code for qiskit.transpiler.passes.utils.cx_direction

# -*- coding: utf-8 -*-

# This code is part of Qiskit.
#
# (C) Copyright IBM 2017, 2018.
#
# This code is licensed under the Apache License, Version 2.0. You may
# obtain a copy of this license in the LICENSE.txt file in the root directory
# of this source tree or at http://www.apache.org/licenses/LICENSE-2.0.
#
# Any modifications or derivative works of this code must retain this
# copyright notice, and modified files need to carry a notice indicating
# that they have been altered from the originals.

"""Rearrange the direction of the cx nodes to match the directed coupling map."""

from math import pi

from qiskit.transpiler.basepasses import TransformationPass
from qiskit.transpiler.exceptions import TranspilerError

from qiskit.circuit import QuantumRegister
from qiskit.dagcircuit import DAGCircuit
from qiskit.circuit.library.standard_gates import U2Gate, CXGate


[docs]class CXDirection(TransformationPass): """Rearrange the direction of the cx nodes to match the directed coupling map. Flip the cx nodes to match the directed coupling map. This pass makes use of the following equivalence:: ---(+)--- --[H]---.---[H]-- | = | ----.---- --[H]--(+)--[H]-- """ def __init__(self, coupling_map): """CXDirection initializer. Args: coupling_map (CouplingMap): Directed graph represented a coupling map. """ super().__init__() self.coupling_map = coupling_map
[docs] def run(self, dag): """Run the CXDirection pass on `dag`. Flips the cx nodes to match the directed coupling map. Modifies the input dag. Args: dag (DAGCircuit): DAG to map. Returns: DAGCircuit: The rearranged dag for the coupling map Raises: TranspilerError: If the circuit cannot be mapped just by flipping the cx nodes. """ cmap_edges = set(self.coupling_map.get_edges()) if len(dag.qregs) > 1: raise TranspilerError('CXDirection expects a single qreg input DAG,' 'but input DAG had qregs: {}.'.format( dag.qregs)) for cnot_node in dag.named_nodes('cx', 'CX'): control = cnot_node.qargs[0] target = cnot_node.qargs[1] physical_q0 = control.index physical_q1 = target.index if self.coupling_map.distance(physical_q0, physical_q1) != 1: raise TranspilerError('The circuit requires a connection between physical ' 'qubits %s and %s' % (physical_q0, physical_q1)) if (physical_q0, physical_q1) not in cmap_edges: # A flip needs to be done # Create the replacement dag and associated register. sub_dag = DAGCircuit() sub_qr = QuantumRegister(2) sub_dag.add_qreg(sub_qr) # Add H gates before sub_dag.apply_operation_back(U2Gate(0, pi), [sub_qr[0]], []) sub_dag.apply_operation_back(U2Gate(0, pi), [sub_qr[1]], []) # Flips the cx sub_dag.apply_operation_back(CXGate(), [sub_qr[1], sub_qr[0]], []) # Add H gates after sub_dag.apply_operation_back(U2Gate(0, pi), [sub_qr[0]], []) sub_dag.apply_operation_back(U2Gate(0, pi), [sub_qr[1]], []) dag.substitute_node_with_dag(cnot_node, sub_dag) return dag