Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Port TwoQubitControlledUDecomposer to rust #13139

Merged
merged 31 commits into from
Nov 13, 2024
Merged
Show file tree
Hide file tree
Changes from 12 commits
Commits
Show all changes
31 commits
Select commit Hold shift + click to select a range
2e3a383
replace calls to TwoQubitWeylDecomposition by calling rust functions
ShellyGarion Sep 12, 2024
7ab59b4
fix failing test
ShellyGarion Sep 12, 2024
cbc9b9d
Merge branch 'main' into two_qubit_decomp
ShellyGarion Oct 8, 2024
285a91d
add code for TwoQubitControlledUDecomposer in rust
ShellyGarion Oct 8, 2024
18000fe
Merge branch 'main' into two_qubit_decomp
ShellyGarion Oct 9, 2024
14c4a56
update try_inverse to inverse
ShellyGarion Oct 9, 2024
eb1b928
fix rust code. add rust docstrings
ShellyGarion Oct 9, 2024
c9b7553
remove python code
ShellyGarion Oct 9, 2024
71caa9e
update unitary in call
ShellyGarion Oct 9, 2024
eb4968b
handle global_phase
ShellyGarion Oct 10, 2024
a2943c9
minor fix
ShellyGarion Oct 10, 2024
7acb09a
fix lint
ShellyGarion Oct 10, 2024
e532c8a
set self.rxx_equivalent_gate
ShellyGarion Oct 15, 2024
3182838
check that rxx_equivalent_gate is a standard gate
ShellyGarion Oct 15, 2024
59ef018
iterating over test_angles and scales
ShellyGarion Oct 15, 2024
1ceb94e
add Vec capacity
ShellyGarion Oct 15, 2024
9ac68b1
remove invert variable
ShellyGarion Oct 15, 2024
6ad3869
move comments from Python to Rust code
ShellyGarion Oct 15, 2024
ebc0e73
remove new_inner function, move code into new
ShellyGarion Oct 15, 2024
02cd73e
call returns a CircuitData
ShellyGarion Oct 17, 2024
71d4b60
Merge remote-tracking branch 'origin/main' into two_qubit_decomp
mtreinish Oct 29, 2024
f46edbe
Support custom gates for decomposition too
mtreinish Oct 29, 2024
2e57608
Remove bare panics
mtreinish Oct 29, 2024
319c6e1
Merge branch 'main' into two_qubit_decomp
mtreinish Oct 29, 2024
7e0c2a2
Undo debug testing for invalid custom gate check
mtreinish Oct 29, 2024
7769833
Fix lint
mtreinish Oct 29, 2024
af17ab8
Fix python lint
mtreinish Oct 30, 2024
06c3da4
add default fidelity value
ShellyGarion Nov 12, 2024
ee3e88e
add more gates for the tests
ShellyGarion Nov 12, 2024
1bb0616
assert that CustomXYGate raises an error
ShellyGarion Nov 12, 2024
8949f7f
update assertion in test
ShellyGarion Nov 13, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
349 changes: 348 additions & 1 deletion crates/accelerate/src/two_qubit_decompose.rs
Original file line number Diff line number Diff line change
Expand Up @@ -54,7 +54,7 @@ use rand_pcg::Pcg64Mcg;
use qiskit_circuit::circuit_data::CircuitData;
use qiskit_circuit::circuit_instruction::OperationFromPython;
use qiskit_circuit::gate_matrix::{CX_GATE, H_GATE, ONE_QUBIT_IDENTITY, SX_GATE, X_GATE};
use qiskit_circuit::operations::{Param, StandardGate};
use qiskit_circuit::operations::{Operation, Param, StandardGate};
use qiskit_circuit::packed_instruction::PackedOperation;
use qiskit_circuit::slice::{PySequenceIndex, SequenceIndex};
use qiskit_circuit::util::{c64, GateArray1Q, GateArray2Q, C_M_ONE, C_ONE, C_ZERO, IM, M_IM};
Expand Down Expand Up @@ -2344,6 +2344,352 @@ pub fn local_equivalence(weyl: PyReadonlyArray1<f64>) -> PyResult<[f64; 3]> {
Ok([g0_equiv + 0., g1_equiv + 0., g2_equiv + 0.])
}

/// invert 1q gate sequence
fn invert_1q_gate(gate: (StandardGate, SmallVec<[f64; 3]>)) -> (StandardGate, SmallVec<[f64; 3]>) {
let gate_params = gate.1.into_iter().map(Param::Float).collect::<Vec<_>>();
let Some(inv_gate) = gate
.0
.inverse(&gate_params) else {panic!()};
let inv_gate_params = inv_gate
.1
.into_iter()
.map(|param| match param {
Param::Float(val) => val,
_ => panic!(),
})
.collect::<SmallVec<_>>();
(inv_gate.0, inv_gate_params)
}

/// invert 2q gate sequence
fn invert_2q_gate(
gate: (Option<StandardGate>, SmallVec<[f64; 3]>, SmallVec<[u8; 2]>),
) -> (StandardGate, SmallVec<[f64; 3]>, SmallVec<[u8; 2]>) {
let Some(inv_gate) = gate
.0
.unwrap()
.inverse(&gate.1.into_iter().map(Param::Float).collect::<Vec<_>>()) else {panic!()};
let inv_gate_params = inv_gate
.1
.into_iter()
.map(|param| match param {
Param::Float(val) => val,
_ => panic!(),
})
.collect::<SmallVec<_>>();
(inv_gate.0, inv_gate_params, gate.2)
}

#[pyclass(module = "qiskit._accelerate.two_qubit_decompose", subclass)]
pub struct TwoQubitControlledUDecomposer {
#[pyo3(get)]
rxx_equivalent_gate: StandardGate,
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do you think we should treat this class as public at all? The python space definition wasn't limited to standard gates, it would support any python space Gate object including a custom defined gate. This will error in the case of a custom PyGate though.

Copy link
Member Author

@ShellyGarion ShellyGarion Oct 15, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Currently, rxx_equivalent_gate could be only a StandardGate since in the code we need to invert it, and there is currently no inverse function for other types, see these lines:

                let circ_c = self.to_rxx_gate(gamma)?;
                ...
                for gate in circ_c.gates.into_iter().rev() {
                    let (inv_gate_name, inv_gate_params, inv_gate_qubits) = invert_2q_gate(gate);

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As for making this class public, I think that it may be useful. Should we do it now or only after it's added to the UnitarySynthesis transpiler pass #13320?

Copy link
Member

@mtreinish mtreinish Oct 29, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'd say lets do it later as the PR is large and intricate enough that we should review the change making it public on its own.

#[pyo3(get)]
scale: f64,
}

const DEFAULT_ATOL: f64 = 1e-12;

/// Decompose two-qubit unitary in terms of a desired
/// :math:`U \sim U_d(\alpha, 0, 0) \sim \text{Ctrl-U}`
/// gate that is locally equivalent to an :class:`.RXXGate`.
impl TwoQubitControlledUDecomposer {
/// Takes an angle and returns the circuit equivalent to an RXXGate with the
/// RXX equivalent gate as the two-qubit unitary.
/// Args:
/// angle: Rotation angle (in this case one of the Weyl parameters a, b, or c)
/// Returns:
/// Circuit: Circuit equivalent to an RXXGate.
/// Raises:
/// QiskitError: If the circuit is not equivalent to an RXXGate.
fn to_rxx_gate(&self, angle: f64) -> PyResult<TwoQubitGateSequence> {
let mat = self
.rxx_equivalent_gate
.matrix(&[Param::Float(self.scale * angle)])
.unwrap();
let decomposer_inv =
TwoQubitWeylDecomposition::new_inner(mat.view(), Some(DEFAULT_FIDELITY), None)?;

let euler_basis = EulerBasis::ZYZ;
let mut target_1q_basis_list = EulerBasisSet::new();
target_1q_basis_list.add_basis(euler_basis);

let mut gates = Vec::new();
ShellyGarion marked this conversation as resolved.
Show resolved Hide resolved
let mut global_phase = -decomposer_inv.global_phase;

let decomp_k1r = decomposer_inv.K1r.view();
let decomp_k2r = decomposer_inv.K2r.view();
let decomp_k1l = decomposer_inv.K1l.view();
let decomp_k2l = decomposer_inv.K2l.view();

let unitary_k1r =
unitary_to_gate_sequence_inner(decomp_k1r, &target_1q_basis_list, 0, None, true, None);
let unitary_k2r =
unitary_to_gate_sequence_inner(decomp_k2r, &target_1q_basis_list, 0, None, true, None);
let unitary_k1l =
unitary_to_gate_sequence_inner(decomp_k1l, &target_1q_basis_list, 0, None, true, None);
let unitary_k2l =
unitary_to_gate_sequence_inner(decomp_k2l, &target_1q_basis_list, 0, None, true, None);

if let Some(unitary_k2r) = unitary_k2r {
global_phase -= unitary_k2r.global_phase;
for gate in unitary_k2r.gates.into_iter().rev() {
let (inv_gate_name, inv_gate_params) = invert_1q_gate(gate);
gates.push((Some(inv_gate_name), inv_gate_params, smallvec![0]));
}
}
if let Some(unitary_k2l) = unitary_k2l {
global_phase -= unitary_k2l.global_phase;
for gate in unitary_k2l.gates.into_iter().rev() {
let (inv_gate_name, inv_gate_params) = invert_1q_gate(gate);
gates.push((Some(inv_gate_name), inv_gate_params, smallvec![1]));
}
}
gates.push((
Some(self.rxx_equivalent_gate),
smallvec![self.scale * angle],
smallvec![0, 1],
));

if let Some(unitary_k1r) = unitary_k1r {
global_phase += unitary_k1r.global_phase;
for gate in unitary_k1r.gates.into_iter().rev() {
let (inv_gate_name, inv_gate_params) = invert_1q_gate(gate);
gates.push((Some(inv_gate_name), inv_gate_params, smallvec![0]));
}
}
if let Some(unitary_k1l) = unitary_k1l {
global_phase += unitary_k1l.global_phase;
for gate in unitary_k1l.gates.into_iter().rev() {
let (inv_gate_name, inv_gate_params) = invert_1q_gate(gate);
gates.push((Some(inv_gate_name), inv_gate_params, smallvec![1]));
}
}

Ok(TwoQubitGateSequence {
gates,
global_phase,
})
}

/// Appends U_d(a, b, c) to the circuit.
fn weyl_gate(
&self,
circ: &mut TwoQubitGateSequence,
target_decomposed: TwoQubitWeylDecomposition,
atol: f64,
) -> PyResult<()> {
let circ_a = self.to_rxx_gate(-2.0 * target_decomposed.a)?;
circ.gates.extend(circ_a.gates);
let mut global_phase = circ_a.global_phase;

if (target_decomposed.b).abs() > atol {
let circ_b = self.to_rxx_gate(-2.0 * target_decomposed.b)?;
global_phase += circ_b.global_phase;
circ.gates
.push((Some(StandardGate::SdgGate), smallvec![], smallvec![0]));
circ.gates
.push((Some(StandardGate::SdgGate), smallvec![], smallvec![1]));
circ.gates.extend(circ_b.gates);
circ.gates
.push((Some(StandardGate::SGate), smallvec![], smallvec![0]));
circ.gates
.push((Some(StandardGate::SGate), smallvec![], smallvec![1]));
}

if (target_decomposed.c).abs() > atol {
let mut gamma = -2.0 * target_decomposed.c;
let mut invert = false;
if gamma > 0.0 {
gamma *= -1.0;
invert = true;
}
let circ_c = self.to_rxx_gate(gamma)?;
if !invert {
global_phase += circ_c.global_phase;
circ.gates
.push((Some(StandardGate::HGate), smallvec![], smallvec![0]));
circ.gates
.push((Some(StandardGate::HGate), smallvec![], smallvec![1]));
circ.gates.extend(circ_c.gates);
circ.gates
.push((Some(StandardGate::HGate), smallvec![], smallvec![0]));
circ.gates
.push((Some(StandardGate::HGate), smallvec![], smallvec![1]));
} else {
// invert the circuit above
global_phase -= circ_c.global_phase;
circ.gates
.push((Some(StandardGate::HGate), smallvec![], smallvec![0]));
circ.gates
.push((Some(StandardGate::HGate), smallvec![], smallvec![1]));
for gate in circ_c.gates.into_iter().rev() {
let (inv_gate_name, inv_gate_params, inv_gate_qubits) = invert_2q_gate(gate);
circ.gates
.push((Some(inv_gate_name), inv_gate_params, inv_gate_qubits));
}
circ.gates
.push((Some(StandardGate::HGate), smallvec![], smallvec![0]));
circ.gates
.push((Some(StandardGate::HGate), smallvec![], smallvec![1]));
}
ShellyGarion marked this conversation as resolved.
Show resolved Hide resolved
}

circ.global_phase = global_phase;
Ok(())
}

/// Returns the Weyl decomposition in circuit form.
/// Note: atol is passed to OneQubitEulerDecomposer.
fn call_inner(
&self,
unitary: ArrayView2<Complex64>,
atol: f64,
) -> PyResult<TwoQubitGateSequence> {
let target_decomposed =
TwoQubitWeylDecomposition::new_inner(unitary, Some(DEFAULT_FIDELITY), None)?;

let euler_basis = EulerBasis::ZYZ;
let mut target_1q_basis_list = EulerBasisSet::new();
target_1q_basis_list.add_basis(euler_basis);

let c1r = target_decomposed.K1r.view();
let c2r = target_decomposed.K2r.view();
let c1l = target_decomposed.K1l.view();
let c2l = target_decomposed.K2l.view();

let unitary_c1r =
unitary_to_gate_sequence_inner(c1r, &target_1q_basis_list, 0, None, true, None);
let unitary_c2r =
unitary_to_gate_sequence_inner(c2r, &target_1q_basis_list, 0, None, true, None);
let unitary_c1l =
unitary_to_gate_sequence_inner(c1l, &target_1q_basis_list, 0, None, true, None);
let unitary_c2l =
unitary_to_gate_sequence_inner(c2l, &target_1q_basis_list, 0, None, true, None);

let mut gates = Vec::new();
ShellyGarion marked this conversation as resolved.
Show resolved Hide resolved
let mut global_phase = target_decomposed.global_phase;

if let Some(unitary_c2r) = unitary_c2r {
global_phase += unitary_c2r.global_phase;
for gate in unitary_c2r.gates.into_iter() {
gates.push((Some(gate.0), gate.1, smallvec![0]));
}
}
if let Some(unitary_c2l) = unitary_c2l {
global_phase += unitary_c2l.global_phase;
for gate in unitary_c2l.gates.into_iter() {
gates.push((Some(gate.0), gate.1, smallvec![1]));
}
}
let mut gates1 = TwoQubitGateSequence {
gates,
global_phase,
};
self.weyl_gate(&mut gates1, target_decomposed, atol)?;
global_phase += gates1.global_phase;

if let Some(unitary_c1r) = unitary_c1r {
global_phase -= unitary_c1r.global_phase;
for gate in unitary_c1r.gates.into_iter() {
gates1.gates.push((Some(gate.0), gate.1, smallvec![0]));
}
}
if let Some(unitary_c1l) = unitary_c1l {
global_phase -= unitary_c1l.global_phase;
for gate in unitary_c1l.gates.into_iter() {
gates1.gates.push((Some(gate.0), gate.1, smallvec![1]));
}
}

gates1.global_phase = global_phase;
Ok(gates1)
}

/// Initialize the KAK decomposition.
/// Args:
/// rxx_equivalent_gate: Gate that is locally equivalent to an :class:`.RXXGate`:
/// :math:`U \sim U_d(\alpha, 0, 0) \sim \text{Ctrl-U}` gate.
/// Raises:
/// QiskitError: If the gate is not locally equivalent to an :class:`.RXXGate`.
fn new_inner(rxx_equivalent_gate: StandardGate) -> PyResult<Self> {
let atol = DEFAULT_ATOL;
let mut scales = Vec::new();
let test_angles = [0.2, 0.3, PI2];

for test_angle in test_angles {
if rxx_equivalent_gate.num_params() != 1 {
return Err(QiskitError::new_err(
"Equivalent gate needs to take exactly 1 angle parameter.",
));
}
let mat = rxx_equivalent_gate
.matrix(&[Param::Float(test_angle)])
.unwrap();
let decomp =
TwoQubitWeylDecomposition::new_inner(mat.view(), Some(DEFAULT_FIDELITY), None)?;

let mat_rxx = StandardGate::RXXGate
.matrix(&[Param::Float(test_angle)])
.unwrap();
let decomposer_rxx = TwoQubitWeylDecomposition::new_inner(
mat_rxx.view(),
None,
Some(Specialization::ControlledEquiv),
)?;
let decomposer_equiv = TwoQubitWeylDecomposition::new_inner(
mat.view(),
None,
Some(Specialization::ControlledEquiv),
)?;
let scale_a = decomposer_rxx.a / decomposer_equiv.a;

if (decomp.a * 2.0 - test_angle / scale_a).abs() > atol {
return Err(QiskitError::new_err(format!(
"The gate {}
is not equivalent to an RXXGate.",
rxx_equivalent_gate.name()
)));
}

scales.push(scale_a);
}
ShellyGarion marked this conversation as resolved.
Show resolved Hide resolved

let scale = scales[0];

// Check that all three tested angles give the same scale
for scale_val in scales.clone().into_iter() {
if !abs_diff_eq!(scale_val, scale, epsilon = atol) {
return Err(QiskitError::new_err(
"Inconsistent scaling parameters in check.",
));
}
}
ShellyGarion marked this conversation as resolved.
Show resolved Hide resolved

Ok(TwoQubitControlledUDecomposer {
scale,
rxx_equivalent_gate,
})
}
}

#[pymethods]
impl TwoQubitControlledUDecomposer {
#[new]
#[pyo3(signature=(rxx_equivalent_gate))]
fn new(rxx_equivalent_gate: StandardGate) -> PyResult<Self> {
TwoQubitControlledUDecomposer::new_inner(rxx_equivalent_gate)
}
ShellyGarion marked this conversation as resolved.
Show resolved Hide resolved
#[pyo3(signature=(unitary, atol))]
fn __call__(
&self,
unitary: PyReadonlyArray2<Complex64>,
atol: f64,
) -> PyResult<TwoQubitGateSequence> {
mtreinish marked this conversation as resolved.
Show resolved Hide resolved
self.call_inner(unitary.as_array(), atol)
}
}

pub fn two_qubit_decompose(m: &Bound<PyModule>) -> PyResult<()> {
m.add_wrapped(wrap_pyfunction!(_num_basis_gates))?;
m.add_wrapped(wrap_pyfunction!(py_decompose_two_qubit_product_gate))?;
Expand All @@ -2356,5 +2702,6 @@ pub fn two_qubit_decompose(m: &Bound<PyModule>) -> PyResult<()> {
m.add_class::<TwoQubitWeylDecomposition>()?;
m.add_class::<Specialization>()?;
m.add_class::<TwoQubitBasisDecomposer>()?;
m.add_class::<TwoQubitControlledUDecomposer>()?;
Ok(())
}
Loading