From 37bce3902fc71151c22a50fb79a92c07be6a28de Mon Sep 17 00:00:00 2001 From: Toshinari Itoko Date: Fri, 27 Jan 2023 21:39:33 +0900 Subject: [PATCH 01/12] Add Clifford.from_matrix --- .../operators/symplectic/clifford.py | 81 +++++++++++++++++++ .../operators/symplectic/clifford_circuits.py | 28 +++++-- 2 files changed, 101 insertions(+), 8 deletions(-) diff --git a/qiskit/quantum_info/operators/symplectic/clifford.py b/qiskit/quantum_info/operators/symplectic/clifford.py index 3773a7b53700..fde66e3c0140 100644 --- a/qiskit/quantum_info/operators/symplectic/clifford.py +++ b/qiskit/quantum_info/operators/symplectic/clifford.py @@ -551,6 +551,21 @@ def to_matrix(self): """Convert operator to Numpy matrix.""" return self.to_operator().data + @classmethod + def from_matrix(cls, matrix): + """Create a Clifford from a unitary matrix. + + Note that this function takes exponentially longer w.r.t. the number of qubits. + + Returns: + Clifford: the Clifford object for the unitary matrix. + + Raises: + QiskitError: if the input instruction is non-Clifford matrix. + """ + tableau = cls._unitary_matrix_to_tableau(matrix) + return cls(tableau) + def to_operator(self): """Convert to an Operator object.""" return Operator(self.to_instruction()) @@ -848,6 +863,72 @@ def _from_label(label): symp[-1] = phase return symp + @staticmethod + def _unitary_matrix_to_tableau(matrix): + # pylint: disable=invalid-name + U = Operator(matrix) + Udg = U.adjoint() + num_qubits = U.num_qubits + # Create full table for a n-qubit Pauli matrix to a row of the tableau + x_table = { + "I": False, + "X": True, + "Z": False, + "Y": True, + } + z_table = { + "I": False, + "X": False, + "Z": True, + "Y": True, + } + matrix_row_pairs = [] + for paulis in itertools.product(["I", "X", "Y", "Z"], repeat=num_qubits): + # positive phase Pauli case + row_p = np.empty(2 * num_qubits + 1, dtype=bool) + phase = False + for i, pauli in enumerate(paulis): + row_p[i] = x_table[pauli] + row_p[i + num_qubits] = z_table[pauli] + phase ^= pauli == "Y" + row_p[-1] = phase + pauli_str = "".join(reversed(paulis)) + pauli_op = Operator.from_label(pauli_str) + matrix_p = ((-1) ** phase) * pauli_op.to_matrix() + matrix_row_pairs.append((matrix_p, row_p)) + # negative phase Pauli case + row_n = np.copy(row_p) + row_n[-1] ^= True + matrix_row_pairs.append(((-1) * matrix_p, row_n)) + + # Construct stabilizer/destabilizer tables + stab = np.empty((num_qubits, 2 * num_qubits + 1), dtype=bool) + for i in range(num_qubits): + label = "I" * (num_qubits - i - 1) + "X" + "I" * i + Xi = Operator.from_label(label).to_matrix() + target = U @ Xi @ Udg + for mat, row in matrix_row_pairs: + if np.allclose(mat, target): + stab[i] = row + break + else: + raise QiskitError("Non-Clifford matrix is not convertible") + + destab = np.empty((num_qubits, 2 * num_qubits + 1), dtype=bool) + for i in range(num_qubits): + label = "I" * (num_qubits - i - 1) + "Z" + "I" * i + Zi = Operator.from_label(label).to_matrix() + target = U @ Zi @ Udg + for mat, row in matrix_row_pairs: + if np.allclose(mat, target): + destab[i] = row + break + else: + raise QiskitError("Non-Clifford matrix is not convertible") + + tableau = np.vstack([stab, destab]) + return tableau + # Update docstrings for API docs generate_apidocs(Clifford) diff --git a/qiskit/quantum_info/operators/symplectic/clifford_circuits.py b/qiskit/quantum_info/operators/symplectic/clifford_circuits.py index 437913034764..57ee130ec1ee 100644 --- a/qiskit/quantum_info/operators/symplectic/clifford_circuits.py +++ b/qiskit/quantum_info/operators/symplectic/clifford_circuits.py @@ -13,8 +13,8 @@ Circuit simulation for the Clifford class. """ -from qiskit.circuit.barrier import Barrier -from qiskit.circuit.delay import Delay +from qiskit.circuit import Barrier, Delay, Gate +from qiskit.circuit.exceptions import CircuitError from qiskit.exceptions import QiskitError @@ -51,7 +51,7 @@ def _append_operation(clifford, operation, qargs=None): Args: clifford (Clifford): the Clifford to update. - operation (Instruction or str): the operation or composite operation to apply. + operation (Instruction or Clifford or str): the operation or composite operation to apply. qargs (list or None): The qubits to apply operation to. Returns: @@ -103,12 +103,24 @@ def _append_operation(clifford, operation, qargs=None): clifford.tableau = composed_clifford.tableau return clifford - # If not a Clifford basis gate we try to unroll the gate and - # raise an exception if unrolling reaches a non-Clifford gate. - # TODO: We could also check u3 params to see if they - # are a single qubit Clifford gate rather than raise an exception. + # If the gate is not in predefined basis gates but with up to 3 qubits, + # we try to construct a Clifford to be appended from its matrix representation. + if isinstance(gate, Gate) and len(qargs) <= 3: + try: + matrix = gate.to_matrix() + try: + gate_cliff = Clifford.from_matrix(matrix) + return _append_operation(clifford, gate_cliff, qargs=qargs) + except QiskitError as err: + raise QiskitError(f"Cannot apply non-Clifford gate: {gate.name}") from err + except TypeError as err: + raise QiskitError(f"Cannot apply {gate.name} gate with unbounded parameters") from err + except CircuitError: + pass + + # If the gate is not directly appendable, we try to unroll the gate with its definition if gate.definition is None: - raise QiskitError(f"Cannot apply Instruction: {gate.name}") + raise QiskitError(f"Cannot apply {gate}") return _append_circuit(clifford, gate.definition, qargs) From 6019306cb71a0c5041b0a0590d9aafea630f42bc Mon Sep 17 00:00:00 2001 From: Toshinari Itoko Date: Mon, 30 Jan 2023 17:41:15 +0900 Subject: [PATCH 02/12] Add tests --- .../operators/symplectic/test_clifford.py | 54 +++++++++++++++++++ 1 file changed, 54 insertions(+) diff --git a/test/python/quantum_info/operators/symplectic/test_clifford.py b/test/python/quantum_info/operators/symplectic/test_clifford.py index 8009b4541d2b..38dd816fc863 100644 --- a/test/python/quantum_info/operators/symplectic/test_clifford.py +++ b/test/python/quantum_info/operators/symplectic/test_clifford.py @@ -21,22 +21,39 @@ from qiskit.circuit import Gate, QuantumCircuit, QuantumRegister from qiskit.circuit.library import ( + CPhaseGate, + CRXGate, + CRYGate, + CRZGate, CXGate, + CYGate, CZGate, + ECRGate, HGate, IGate, + RXGate, + RYGate, + RZGate, + RXXGate, + RYYGate, + RZZGate, + RZXGate, SdgGate, SGate, SwapGate, XGate, + XXMinusYYGate, + XXPlusYYGate, YGate, ZGate, + iSwapGate, LinearFunction, PauliGate, ) from qiskit.exceptions import QiskitError from qiskit.quantum_info import random_clifford from qiskit.quantum_info.operators import Clifford, Operator +from qiskit.quantum_info.operators.predicates import matrix_equal from qiskit.quantum_info.operators.symplectic.clifford_circuits import _append_operation from qiskit.synthesis.clifford import ( synth_clifford_full, @@ -942,6 +959,43 @@ def visualize_does_not_throw_error(self): clifford = random_clifford(3, seed=0) print(clifford) + @combine(num_qubits=[1, 2, 3, 4]) + def test_from_matrix_round_trip(self, num_qubits): + """Test round trip conversion to and from matrix""" + for i in range(10): + expected = random_clifford(num_qubits, seed=42 + i) + actual = Clifford.from_matrix(expected.to_matrix()) + self.assertEqual(expected, actual) + + @combine( + gate=[ + RXGate(theta=np.pi / 2), + RYGate(theta=np.pi / 2), + RZGate(phi=np.pi / 2), + CPhaseGate(theta=np.pi), + CRXGate(theta=np.pi), + CRYGate(theta=np.pi), + CRZGate(theta=np.pi), + CXGate(), + CYGate(), + CZGate(), + ECRGate(), + RXXGate(theta=np.pi / 2), + RYYGate(theta=np.pi / 2), + RZZGate(theta=np.pi / 2), + RZXGate(theta=np.pi / 2), + SwapGate(), + iSwapGate(), + XXMinusYYGate(theta=np.pi), + XXPlusYYGate(theta=-np.pi), + ] + ) + def test_create_from_gates(self, gate): + """Test if matrix of Clifford created from gate equals the gate matrix up to global phase""" + self.assertTrue( + matrix_equal(Clifford(gate).to_matrix(), gate.to_matrix(), ignore_phase=True) + ) + if __name__ == "__main__": unittest.main() From dba2eff9c1f56adb7374fdf6b4f4aca4e1508514 Mon Sep 17 00:00:00 2001 From: Toshinari Itoko Date: Mon, 30 Jan 2023 18:41:02 +0900 Subject: [PATCH 03/12] Add reno --- .../notes/add-clifford-from-matrix-3184822cc559e0b7.yaml | 8 ++++++++ 1 file changed, 8 insertions(+) create mode 100644 releasenotes/notes/add-clifford-from-matrix-3184822cc559e0b7.yaml diff --git a/releasenotes/notes/add-clifford-from-matrix-3184822cc559e0b7.yaml b/releasenotes/notes/add-clifford-from-matrix-3184822cc559e0b7.yaml new file mode 100644 index 000000000000..2285266a488f --- /dev/null +++ b/releasenotes/notes/add-clifford-from-matrix-3184822cc559e0b7.yaml @@ -0,0 +1,8 @@ +--- +features: + - | + Added :meth:`.Clifford.from_matrix` method + that creates a ``Clifford`` object from its unitary matrix. + - | + The constructor of :class:`.Clifford` now can take any Clifford gate object up to 3 qubits, + such as ``Rz(pi/2)``, ``iSwapGate()`` and ``ECRGate()``, which were not convertible before. From bad10dec64c2d6a72f27f3a6125bc00e15ca362d Mon Sep 17 00:00:00 2001 From: Toshinari Itoko Date: Wed, 15 Feb 2023 20:17:56 +0900 Subject: [PATCH 04/12] Faster Clifford.from_matrix O(16^n)->O(4^n) --- .../operators/symplectic/clifford.py | 126 +++++++++++------- 1 file changed, 76 insertions(+), 50 deletions(-) diff --git a/qiskit/quantum_info/operators/symplectic/clifford.py b/qiskit/quantum_info/operators/symplectic/clifford.py index fde66e3c0140..708cc48ec266 100644 --- a/qiskit/quantum_info/operators/symplectic/clifford.py +++ b/qiskit/quantum_info/operators/symplectic/clifford.py @@ -12,6 +12,7 @@ """ Clifford operator class. """ +import functools import itertools import re @@ -564,6 +565,8 @@ def from_matrix(cls, matrix): QiskitError: if the input instruction is non-Clifford matrix. """ tableau = cls._unitary_matrix_to_tableau(matrix) + if tableau is None: + raise QiskitError("Non-Clifford matrix is not convertible") return cls(tableau) def to_operator(self): @@ -863,68 +866,91 @@ def _from_label(label): symp[-1] = phase return symp + @staticmethod + def _pauli_matrix_to_row(mat, num_qubits): + """Generate a binary vector (a row of tableau representation) from a Pauli matrix. + Return None if the non-Pauli matrix is supplied.""" + # pylint: disable=too-many-return-statements + + def find_one_index(x, decimals=6): + indices = np.where(np.round(np.abs(x), decimals) == 1) + return indices[0][0] if len(indices[0]) == 1 else None + + def bitvector(n, num_bits): + return np.array([int(digit) for digit in format(n, f"0{num_bits}b")], dtype=bool)[::-1] + + # compute x-bits + xint = find_one_index(mat[0, :]) + if xint is None: + return None + xbits = bitvector(xint, num_qubits) + + # extract non-zero elements from matrix (rounded to 1, -1, 1j or -1j) + entries = np.empty(len(mat), dtype=complex) + for i, row in enumerate(mat): + index = find_one_index(row) + if index is None: + return None + expected = xint ^ i + if index != expected: + return None + entries[i] = np.round(mat[i, index]) + + # compute z-bits + zbits = np.empty(num_qubits, dtype=bool) + for k in range(num_qubits): + sign = np.round(entries[2**k] / entries[0]) + if sign == 1: + zbits[k] = False + elif sign == -1: + zbits[k] = True + else: + return None + + # compute phase + phase = None + num_y = sum(xbits & zbits) + positive_phase = (-1j) ** num_y + if entries[0] == positive_phase: + phase = False + elif entries[0] == -1 * positive_phase: + phase = True + if phase is None: + return None + + # validate all non-zero elements + coef = ((-1) ** phase) * positive_phase + ivec, zvec = np.ones(2), np.array([1, -1]) + expected = coef * functools.reduce(np.kron, [zvec if z else ivec for z in zbits[::-1]]) + if not np.allclose(entries, expected): + return None + + return np.hstack([xbits, zbits, phase]) + @staticmethod def _unitary_matrix_to_tableau(matrix): # pylint: disable=invalid-name - U = Operator(matrix) - Udg = U.adjoint() - num_qubits = U.num_qubits - # Create full table for a n-qubit Pauli matrix to a row of the tableau - x_table = { - "I": False, - "X": True, - "Z": False, - "Y": True, - } - z_table = { - "I": False, - "X": False, - "Z": True, - "Y": True, - } - matrix_row_pairs = [] - for paulis in itertools.product(["I", "X", "Y", "Z"], repeat=num_qubits): - # positive phase Pauli case - row_p = np.empty(2 * num_qubits + 1, dtype=bool) - phase = False - for i, pauli in enumerate(paulis): - row_p[i] = x_table[pauli] - row_p[i + num_qubits] = z_table[pauli] - phase ^= pauli == "Y" - row_p[-1] = phase - pauli_str = "".join(reversed(paulis)) - pauli_op = Operator.from_label(pauli_str) - matrix_p = ((-1) ** phase) * pauli_op.to_matrix() - matrix_row_pairs.append((matrix_p, row_p)) - # negative phase Pauli case - row_n = np.copy(row_p) - row_n[-1] ^= True - matrix_row_pairs.append(((-1) * matrix_p, row_n)) - - # Construct stabilizer/destabilizer tables + num_qubits = int(np.log2(len(matrix))) + stab = np.empty((num_qubits, 2 * num_qubits + 1), dtype=bool) for i in range(num_qubits): label = "I" * (num_qubits - i - 1) + "X" + "I" * i Xi = Operator.from_label(label).to_matrix() - target = U @ Xi @ Udg - for mat, row in matrix_row_pairs: - if np.allclose(mat, target): - stab[i] = row - break - else: - raise QiskitError("Non-Clifford matrix is not convertible") + target = matrix @ Xi @ np.conj(matrix).T + row = Clifford._pauli_matrix_to_row(target, num_qubits) + if row is None: + return None + stab[i] = row destab = np.empty((num_qubits, 2 * num_qubits + 1), dtype=bool) for i in range(num_qubits): label = "I" * (num_qubits - i - 1) + "Z" + "I" * i Zi = Operator.from_label(label).to_matrix() - target = U @ Zi @ Udg - for mat, row in matrix_row_pairs: - if np.allclose(mat, target): - destab[i] = row - break - else: - raise QiskitError("Non-Clifford matrix is not convertible") + target = matrix @ Zi @ np.conj(matrix).T + row = Clifford._pauli_matrix_to_row(target, num_qubits) + if row is None: + return None + destab[i] = row tableau = np.vstack([stab, destab]) return tableau From 7a0c3732fa1844f2ab84eb226c9ceb34c447d856 Mon Sep 17 00:00:00 2001 From: Toshinari Itoko Date: Wed, 15 Feb 2023 20:18:16 +0900 Subject: [PATCH 05/12] Add infinite recursion test case --- .../operators/symplectic/test_clifford.py | 34 +++++++++++++++++++ 1 file changed, 34 insertions(+) diff --git a/test/python/quantum_info/operators/symplectic/test_clifford.py b/test/python/quantum_info/operators/symplectic/test_clifford.py index 38dd816fc863..aec689098098 100644 --- a/test/python/quantum_info/operators/symplectic/test_clifford.py +++ b/test/python/quantum_info/operators/symplectic/test_clifford.py @@ -40,6 +40,7 @@ RZXGate, SdgGate, SGate, + SXGate, SwapGate, XGate, XXMinusYYGate, @@ -511,6 +512,39 @@ def test_from_circuit_with_all_types(self): expected_clifford = Clifford.from_dict(expected_clifford_dict) self.assertEqual(combined_clifford, expected_clifford) + def test_from_gate_with_cyclic_definition(self): + """Test if a Clifford can be created from gate with cyclic definition""" + + class MyHGate(HGate): + """Custom HGate class for test""" + + def __init__(self): + super().__init__() + self.name = "my_h" + + def _define(self): + qc = QuantumCircuit(1, name=self.name) + qc.s(0) + qc.append(MySXGate(), [0]) + qc.s(0) + self.definition = qc + + class MySXGate(SXGate): + """Custom SXGate class for test""" + + def __init__(self): + super().__init__() + self.name = "my_sx" + + def _define(self): + qc = QuantumCircuit(1, name=self.name) + qc.sdg(0) + qc.append(MyHGate(), [0]) + qc.sdg(0) + self.definition = qc + + Clifford(MyHGate()) + @ddt class TestCliffordSynthesis(QiskitTestCase): From b64d969a7c9da839f644fa162848134557dcf5f1 Mon Sep 17 00:00:00 2001 From: Toshinari Itoko Date: Fri, 17 Feb 2023 20:39:34 +0900 Subject: [PATCH 06/12] Change to try append with definition first --- .../operators/symplectic/clifford.py | 6 +-- .../operators/symplectic/clifford_circuits.py | 43 +++++++++++++------ 2 files changed, 32 insertions(+), 17 deletions(-) diff --git a/qiskit/quantum_info/operators/symplectic/clifford.py b/qiskit/quantum_info/operators/symplectic/clifford.py index 708cc48ec266..9019e787bda0 100644 --- a/qiskit/quantum_info/operators/symplectic/clifford.py +++ b/qiskit/quantum_info/operators/symplectic/clifford.py @@ -621,9 +621,9 @@ def from_circuit(circuit): # Initialize an identity Clifford clifford = Clifford(np.eye(2 * circuit.num_qubits), validate=False) if isinstance(circuit, QuantumCircuit): - _append_circuit(clifford, circuit) + clifford = _append_circuit(clifford, circuit) else: - _append_operation(clifford, circuit) + clifford = _append_operation(clifford, circuit) return clifford @staticmethod @@ -679,7 +679,7 @@ def from_label(label): num_qubits = len(label) op = Clifford(np.eye(2 * num_qubits, dtype=bool)) for qubit, char in enumerate(reversed(label)): - _append_operation(op, label_gates[char], qargs=[qubit]) + op = _append_operation(op, label_gates[char], qargs=[qubit]) return op def to_labels(self, array=False, mode="B"): diff --git a/qiskit/quantum_info/operators/symplectic/clifford_circuits.py b/qiskit/quantum_info/operators/symplectic/clifford_circuits.py index 57ee130ec1ee..b12699dc1403 100644 --- a/qiskit/quantum_info/operators/symplectic/clifford_circuits.py +++ b/qiskit/quantum_info/operators/symplectic/clifford_circuits.py @@ -12,19 +12,21 @@ """ Circuit simulation for the Clifford class. """ +import copy from qiskit.circuit import Barrier, Delay, Gate from qiskit.circuit.exceptions import CircuitError from qiskit.exceptions import QiskitError -def _append_circuit(clifford, circuit, qargs=None): +def _append_circuit(clifford, circuit, qargs=None, recursion_depth=0): """Update Clifford inplace by applying a Clifford circuit. Args: - clifford (Clifford): the Clifford to update. - circuit (QuantumCircuit): the circuit to apply. + clifford (Clifford): The Clifford to update. + circuit (QuantumCircuit): The circuit to apply. qargs (list or None): The qubits to apply circuit to. + recursion_depth (int): The depth of mutual recursion with _append_operation Returns: Clifford: the updated Clifford. @@ -42,24 +44,26 @@ def _append_circuit(clifford, circuit, qargs=None): ) # Get the integer position of the flat register new_qubits = [qargs[circuit.find_bit(bit).index] for bit in instruction.qubits] - _append_operation(clifford, instruction.operation, new_qubits) + clifford = _append_operation(clifford, instruction.operation, new_qubits, recursion_depth) return clifford -def _append_operation(clifford, operation, qargs=None): +def _append_operation(clifford, operation, qargs=None, recursion_depth=0): """Update Clifford inplace by applying a Clifford operation. Args: - clifford (Clifford): the Clifford to update. - operation (Instruction or Clifford or str): the operation or composite operation to apply. + clifford (Clifford): The Clifford to update. + operation (Instruction or Clifford or str): The operation or composite operation to apply. qargs (list or None): The qubits to apply operation to. + recursion_depth (int): The depth of mutual recursion with _append_circuit Returns: Clifford: the updated Clifford. Raises: - QiskitError: if input operation cannot be decomposed into Clifford operations. + QiskitError: if input operation cannot be converted into Clifford operations. """ + # pylint: disable=too-many-return-statements if isinstance(operation, (Barrier, Delay)): return clifford @@ -103,7 +107,22 @@ def _append_operation(clifford, operation, qargs=None): clifford.tableau = composed_clifford.tableau return clifford - # If the gate is not in predefined basis gates but with up to 3 qubits, + # If the gate is not directly appendable, we try to unroll the gate with its definition. + # This succeeds only if the gate has all-Clifford definition (decomposition). + # If fails, we need to restore the clifford that was before attempting to unroll and append. + if gate.definition is not None: + if recursion_depth > 0: + return _append_circuit(clifford, gate.definition, qargs, recursion_depth + 1) + else: # recursion_depth == 0 + # clifford may be updated in _append_circuit + org_clifford = copy.deepcopy(clifford) + try: + return _append_circuit(clifford, gate.definition, qargs, 1) + except (QiskitError, RecursionError): + # discard incompletely updated clifford and continue + clifford = org_clifford + + # As a final attempt, if the gate is up to 3 qubits, # we try to construct a Clifford to be appended from its matrix representation. if isinstance(gate, Gate) and len(qargs) <= 3: try: @@ -118,11 +137,7 @@ def _append_operation(clifford, operation, qargs=None): except CircuitError: pass - # If the gate is not directly appendable, we try to unroll the gate with its definition - if gate.definition is None: - raise QiskitError(f"Cannot apply {gate}") - - return _append_circuit(clifford, gate.definition, qargs) + raise QiskitError(f"Cannot apply {gate}") # --------------------------------------------------------------------- From c4bc00a4c8c78ea468b3715d900c25bf4176b41d Mon Sep 17 00:00:00 2001 From: Toshinari Itoko Date: Fri, 17 Feb 2023 21:44:56 +0900 Subject: [PATCH 07/12] Add u gate handling for speed --- .../operators/symplectic/clifford_circuits.py | 56 +++++++++++++++++++ 1 file changed, 56 insertions(+) diff --git a/qiskit/quantum_info/operators/symplectic/clifford_circuits.py b/qiskit/quantum_info/operators/symplectic/clifford_circuits.py index b12699dc1403..209a99f70d69 100644 --- a/qiskit/quantum_info/operators/symplectic/clifford_circuits.py +++ b/qiskit/quantum_info/operators/symplectic/clifford_circuits.py @@ -13,6 +13,7 @@ Circuit simulation for the Clifford class. """ import copy +import numpy as np from qiskit.circuit import Barrier, Delay, Gate from qiskit.circuit.exceptions import CircuitError @@ -95,6 +96,28 @@ def _append_operation(clifford, operation, qargs=None, recursion_depth=0): raise QiskitError("Invalid qubits for 2-qubit gate.") return _BASIS_2Q[name](clifford, qargs[0], qargs[1]) + # If u gate, check if it is a Clifford, and if so, apply it + if isinstance(gate, Gate) and name == "u" and len(qargs) == 1: + try: + theta, phi, lambd = tuple(_n_half_pis(par) for par in gate.params) + except ValueError as err: + raise QiskitError("U gate angles must be multiples of pi/2 to be a Clifford") from err + if theta == 0: + clifford = _append_rz(clifford, qargs[0], lambd + phi) + elif theta == 1: + clifford = _append_rz(clifford, qargs[0], lambd - 2) + clifford = _append_h(clifford, qargs[0]) + clifford = _append_rz(clifford, qargs[0], phi) + elif theta == 2: + clifford = _append_rz(clifford, qargs[0], lambd - 1) + clifford = _append_x(clifford, qargs[0]) + clifford = _append_rz(clifford, qargs[0], phi + 1) + elif theta == 3: + clifford = _append_rz(clifford, qargs[0], lambd) + clifford = _append_h(clifford, qargs[0]) + clifford = _append_rz(clifford, qargs[0], phi + 2) + return clifford + # If gate is a Clifford, we can either unroll the gate using the "to_circuit" # method, or we can compose the Cliffords directly. Experimentally, for large # cliffords the second method is considerably faster. @@ -140,9 +163,42 @@ def _append_operation(clifford, operation, qargs=None, recursion_depth=0): raise QiskitError(f"Cannot apply {gate}") +def _n_half_pis(param) -> int: + try: + param = float(param) + except TypeError as err: + raise ValueError(f"{param} is not bounded") from err + + epsilon = (abs(param) + 0.5 * 1e-10) % (np.pi / 2) + if epsilon < 1e-10: + multiple = int(np.round(param / (np.pi / 2))) + return multiple % 4 + + raise ValueError(f"{param} is not to a multiple of pi/2") + + # --------------------------------------------------------------------- # Helper functions for applying basis gates # --------------------------------------------------------------------- +def _append_rz(clifford, qubit, multiple): + """Apply an Rz gate to a Clifford. + + Args: + clifford (Clifford): a Clifford. + qubit (int): gate qubit index. + multiple (int): z-rotation angle in a multiple of pi/2 + + Returns: + Clifford: the updated Clifford. + """ + if multiple % 4 == 1: + return _append_s(clifford, qubit) + if multiple % 4 == 2: + return _append_z(clifford, qubit) + if multiple % 4 == 3: + return _append_sdg(clifford, qubit) + + return clifford def _append_i(clifford, qubit): From 8ee86cd02d37066cc1d7b28dcc6af9ec33ef7a6c Mon Sep 17 00:00:00 2001 From: Toshinari Itoko Date: Wed, 22 Feb 2023 10:33:14 +0900 Subject: [PATCH 08/12] Improve implematation following review comments --- .../operators/symplectic/clifford_circuits.py | 25 ++++++++----------- 1 file changed, 11 insertions(+), 14 deletions(-) diff --git a/qiskit/quantum_info/operators/symplectic/clifford_circuits.py b/qiskit/quantum_info/operators/symplectic/clifford_circuits.py index 209a99f70d69..92e3980a0a1c 100644 --- a/qiskit/quantum_info/operators/symplectic/clifford_circuits.py +++ b/qiskit/quantum_info/operators/symplectic/clifford_circuits.py @@ -150,15 +150,14 @@ def _append_operation(clifford, operation, qargs=None, recursion_depth=0): if isinstance(gate, Gate) and len(qargs) <= 3: try: matrix = gate.to_matrix() - try: - gate_cliff = Clifford.from_matrix(matrix) - return _append_operation(clifford, gate_cliff, qargs=qargs) - except QiskitError as err: - raise QiskitError(f"Cannot apply non-Clifford gate: {gate.name}") from err + gate_cliff = Clifford.from_matrix(matrix) + return _append_operation(clifford, gate_cliff, qargs=qargs) except TypeError as err: raise QiskitError(f"Cannot apply {gate.name} gate with unbounded parameters") from err - except CircuitError: - pass + except CircuitError as err: + raise QiskitError(f"Cannot apply {gate.name} gate without to_matrix defined") from err + except QiskitError as err: + raise QiskitError(f"Cannot apply non-Clifford gate: {gate.name}") from err raise QiskitError(f"Cannot apply {gate}") @@ -166,15 +165,13 @@ def _append_operation(clifford, operation, qargs=None, recursion_depth=0): def _n_half_pis(param) -> int: try: param = float(param) - except TypeError as err: - raise ValueError(f"{param} is not bounded") from err - - epsilon = (abs(param) + 0.5 * 1e-10) % (np.pi / 2) - if epsilon < 1e-10: + epsilon = (abs(param) + 0.5 * 1e-10) % (np.pi / 2) + if epsilon > 1e-10: + raise ValueError(f"{param} is not to a multiple of pi/2") multiple = int(np.round(param / (np.pi / 2))) return multiple % 4 - - raise ValueError(f"{param} is not to a multiple of pi/2") + except TypeError as err: + raise ValueError(f"{param} is not bounded") from err # --------------------------------------------------------------------- From bbc15356187c66e22b87fc3f1cd059758aa5f555 Mon Sep 17 00:00:00 2001 From: Toshinari Itoko Date: Wed, 22 Feb 2023 11:46:06 +0900 Subject: [PATCH 09/12] Add Clifford.from_operator --- .../operators/symplectic/clifford.py | 27 +++++++++++++++++-- ...clifford-from-matrix-3184822cc559e0b7.yaml | 4 +-- .../operators/symplectic/test_clifford.py | 8 ++++++ 3 files changed, 35 insertions(+), 4 deletions(-) diff --git a/qiskit/quantum_info/operators/symplectic/clifford.py b/qiskit/quantum_info/operators/symplectic/clifford.py index 9019e787bda0..4d59f2fe025c 100644 --- a/qiskit/quantum_info/operators/symplectic/clifford.py +++ b/qiskit/quantum_info/operators/symplectic/clifford.py @@ -556,13 +556,16 @@ def to_matrix(self): def from_matrix(cls, matrix): """Create a Clifford from a unitary matrix. - Note that this function takes exponentially longer w.r.t. the number of qubits. + Note that this function takes exponentially long time w.r.t. the number of qubits. + + Args: + matrix (np.array): A unitary matrix representing a Clifford to be converted. Returns: Clifford: the Clifford object for the unitary matrix. Raises: - QiskitError: if the input instruction is non-Clifford matrix. + QiskitError: if the input is not a Clifford matrix. """ tableau = cls._unitary_matrix_to_tableau(matrix) if tableau is None: @@ -573,6 +576,26 @@ def to_operator(self): """Convert to an Operator object.""" return Operator(self.to_instruction()) + @classmethod + def from_operator(cls, operator): + """Create a Clifford from a operator. + + Note that this function takes exponentially long time w.r.t. the number of qubits. + + Args: + operator (Operator): An operator representing a Clifford to be converted. + + Returns: + Clifford: the Clifford object for the operator. + + Raises: + QiskitError: if the input is not a Clifford operator. + """ + tableau = cls._unitary_matrix_to_tableau(operator.to_matrix()) + if tableau is None: + raise QiskitError("Non-Clifford operator is not convertible") + return cls(tableau) + def to_circuit(self): """Return a QuantumCircuit implementing the Clifford. diff --git a/releasenotes/notes/add-clifford-from-matrix-3184822cc559e0b7.yaml b/releasenotes/notes/add-clifford-from-matrix-3184822cc559e0b7.yaml index 2285266a488f..8ecbbe6f9557 100644 --- a/releasenotes/notes/add-clifford-from-matrix-3184822cc559e0b7.yaml +++ b/releasenotes/notes/add-clifford-from-matrix-3184822cc559e0b7.yaml @@ -1,8 +1,8 @@ --- features: - | - Added :meth:`.Clifford.from_matrix` method - that creates a ``Clifford`` object from its unitary matrix. + Added :meth:`.Clifford.from_matrix` and :meth:`.Clifford.from_operator` method that + creates a ``Clifford`` object from its unitary matrix and operator representation respectively. - | The constructor of :class:`.Clifford` now can take any Clifford gate object up to 3 qubits, such as ``Rz(pi/2)``, ``iSwapGate()`` and ``ECRGate()``, which were not convertible before. diff --git a/test/python/quantum_info/operators/symplectic/test_clifford.py b/test/python/quantum_info/operators/symplectic/test_clifford.py index aec689098098..cf0c250ed034 100644 --- a/test/python/quantum_info/operators/symplectic/test_clifford.py +++ b/test/python/quantum_info/operators/symplectic/test_clifford.py @@ -1001,6 +1001,14 @@ def test_from_matrix_round_trip(self, num_qubits): actual = Clifford.from_matrix(expected.to_matrix()) self.assertEqual(expected, actual) + @combine(num_qubits=[1, 2, 3, 4]) + def test_from_operator_round_trip(self, num_qubits): + """Test round trip conversion to and from operator""" + for i in range(10): + expected = random_clifford(num_qubits, seed=777 + i) + actual = Clifford.from_operator(expected.to_operator()) + self.assertEqual(expected, actual) + @combine( gate=[ RXGate(theta=np.pi / 2), From 96af80bfbb0b439af59c64aab930c6c2420a7c7a Mon Sep 17 00:00:00 2001 From: Toshinari Itoko Date: Fri, 24 Feb 2023 00:15:31 +0900 Subject: [PATCH 10/12] Update reno --- .../notes/add-clifford-from-matrix-3184822cc559e0b7.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/releasenotes/notes/add-clifford-from-matrix-3184822cc559e0b7.yaml b/releasenotes/notes/add-clifford-from-matrix-3184822cc559e0b7.yaml index 8ecbbe6f9557..925092e76955 100644 --- a/releasenotes/notes/add-clifford-from-matrix-3184822cc559e0b7.yaml +++ b/releasenotes/notes/add-clifford-from-matrix-3184822cc559e0b7.yaml @@ -5,4 +5,4 @@ features: creates a ``Clifford`` object from its unitary matrix and operator representation respectively. - | The constructor of :class:`.Clifford` now can take any Clifford gate object up to 3 qubits, - such as ``Rz(pi/2)``, ``iSwapGate()`` and ``ECRGate()``, which were not convertible before. + including parameterized gates such as ``Rz(pi/2)``, which were not convertible before. From c34d9ca70aacce76b2ad1f03d6c10e3f7c3dc27b Mon Sep 17 00:00:00 2001 From: Toshinari Itoko Date: Fri, 24 Feb 2023 10:58:14 +0900 Subject: [PATCH 11/12] Lint --- .../quantum_info/operators/symplectic/test_clifford.py | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/test/python/quantum_info/operators/symplectic/test_clifford.py b/test/python/quantum_info/operators/symplectic/test_clifford.py index 5505d1d0ae26..4c162c8753cc 100644 --- a/test/python/quantum_info/operators/symplectic/test_clifford.py +++ b/test/python/quantum_info/operators/symplectic/test_clifford.py @@ -592,7 +592,8 @@ def _define(self): class TestCliffordSynthesis(QiskitTestCase): """Test Clifford synthesis methods.""" - def _cliffords_1q(self): + @staticmethod + def _cliffords_1q(): clifford_dicts = [ {"stabilizer": ["+Z"], "destabilizer": ["-X"]}, {"stabilizer": ["-Z"], "destabilizer": ["+X"]}, @@ -1036,7 +1037,7 @@ def test_instruction_name(self, num_qubits): clifford = random_clifford(num_qubits, seed=777) self.assertEqual(clifford.to_instruction().name, str(clifford)) - def visualize_does_not_throw_error(self): + def test_visualize_does_not_throw_error(self): """Test to verify that drawing Clifford does not throw an error""" # An error may be thrown if visualization code calls op.condition instead # of getattr(op, "condition", None) From bf028fcd1e6331878f4660d4f2a3f10d623c6421 Mon Sep 17 00:00:00 2001 From: Toshinari Itoko Date: Mon, 27 Feb 2023 14:43:48 +0900 Subject: [PATCH 12/12] more accurate reno --- .../notes/add-clifford-from-matrix-3184822cc559e0b7.yaml | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/releasenotes/notes/add-clifford-from-matrix-3184822cc559e0b7.yaml b/releasenotes/notes/add-clifford-from-matrix-3184822cc559e0b7.yaml index 925092e76955..fd4322d215d6 100644 --- a/releasenotes/notes/add-clifford-from-matrix-3184822cc559e0b7.yaml +++ b/releasenotes/notes/add-clifford-from-matrix-3184822cc559e0b7.yaml @@ -4,5 +4,6 @@ features: Added :meth:`.Clifford.from_matrix` and :meth:`.Clifford.from_operator` method that creates a ``Clifford`` object from its unitary matrix and operator representation respectively. - | - The constructor of :class:`.Clifford` now can take any Clifford gate object up to 3 qubits, + The constructor of :class:`.Clifford` now can take any Clifford gate object up to 3 qubits + as long it supports :meth:`to_matrix` method, including parameterized gates such as ``Rz(pi/2)``, which were not convertible before.