Source code for piel.tools.qutip.unitary
import jax.numpy as jnp
import numpy as np
from typing import Optional
import qutip
__all__ = [
"standard_s_parameters_to_qutip_qobj",
"verify_matrix_is_unitary",
"subunitary_selection_on_range",
"subunitary_selection_on_index",
]
def matrix_to_qutip_qobj(
s_parameters_standard_matrix: jnp.ndarray,
):
"""
This function converts the calculated S-parameters into a standard Unitary matrix topology so that the shape and
dimensions of the matrix can be observed.
I think this means we need to transpose the output of the filtered sax SDense matrix to map it to a QuTip matrix.
Note that the documentation and formatting of the standard `sax` mapping to a S-parameter standard notation is
already in described in piel/piel/sax/utils.py.
From this stage we can implement a ``QObj`` matrix accordingly and perform simulations accordingly. https://qutip.org/docs/latest/guide/qip/qip-basics.html#unitaries
For example, a ``qutip`` representation of an s-gate gate would be:
..code-block::
import numpy as np
import qutip
# S-Gate
s_gate_matrix = np.array([[1., 0], [0., 1.j]])
s_gate = qutip.Qobj(mat, dims=[[2], [2]])
In mathematical notation, this S-gate would be written as:
..math::
S = \\begin{bmatrix}
1 & 0 \\\\
0 & i \\\\
\\end{bmatrix}
Args:
s_parameters_standard_matrix (nso.ndarray): A dictionary of S-parameters in the form of a SDict from `sax`.
Returns:
qobj_unitary (qutip.Qobj): A QuTip QObj representation of the S-parameters in a unitary matrix.
"""
s_parameter_standard_matrix_numpy = np.asarray(s_parameters_standard_matrix)
qobj_unitary = qutip.Qobj(s_parameter_standard_matrix_numpy)
return qobj_unitary
[docs]def subunitary_selection_on_index(
unitary_matrix: jnp.ndarray,
rows_index: jnp.ndarray | tuple,
columns_index: jnp.ndarray | tuple,
):
"""
This function returns a unitary between the indexes selected, and verifies the indexes are valid by checking that
the output matrix is also a unitary.
TODO implement validation of a 2D matrix.
"""
if type(rows_index) is tuple:
rows_index = jnp.asarray(rows_index)
if type(columns_index) is tuple:
rows_index = jnp.asarray(columns_index)
unitary_matrix_row_selection = unitary_matrix.at[rows_index, :].get()
unitary_matrix_row_column_selection = unitary_matrix_row_selection.at[
:, columns_index
].get()
return unitary_matrix_row_column_selection
[docs]def subunitary_selection_on_range(
unitary_matrix: jnp.ndarray,
stop_index: tuple,
start_index: Optional[tuple] = (0, 0),
):
"""
This function returns a unitary between the indexes selected, and verifies the indexes are valid by checking that
the output matrix is also a unitary.
TODO implement validation of a 2D matrix.
"""
start_row = start_index[0]
end_row = stop_index[0]
start_column = start_index[1]
end_column = stop_index[1]
column_range = jnp.arange(start_column, end_column + 1)
row_range = jnp.arange(start_row, end_row + 1)
unitary_matrix_row_column_selection = subunitary_selection_on_index(
unitary_matrix=unitary_matrix,
rows_index=row_range,
columns_index=column_range,
)
return unitary_matrix_row_column_selection
[docs]def verify_matrix_is_unitary(matrix: jnp.ndarray) -> bool:
"""
Verify that the matrix is unitary.
Args:
matrix (jnp.ndarray): The matrix to verify.
Returns:
bool: True if the matrix is unitary, False otherwise.
"""
qobj = matrix_to_qutip_qobj(matrix)
return qobj.check_isunitary()
standard_s_parameters_to_qutip_qobj = matrix_to_qutip_qobj