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

feat: add U and GPhase gates #799

Merged
merged 55 commits into from
Dec 20, 2023
Merged
Show file tree
Hide file tree
Changes from 5 commits
Commits
Show all changes
55 commits
Select commit Hold shift + click to select a range
d784d88
Add U gate
jcjaskula-aws Nov 14, 2023
9633a2a
modification according to feedback
jcjaskula-aws Nov 14, 2023
e68e78c
fix linters
jcjaskula-aws Nov 14, 2023
0c2c9e8
clean commented code
jcjaskula-aws Nov 14, 2023
c3044cb
first version of gphase
jcjaskula-aws Nov 15, 2023
d6d6886
handle drawing and control global phase
jcjaskula-aws Nov 15, 2023
388367c
Adding a global phase attribute
jcjaskula-aws Nov 17, 2023
7a52eb8
add global phase to circuit unitary
jcjaskula-aws Nov 18, 2023
d8917f3
Merge branch 'main' into jcjaskula-aws/add_u_gate
jcjaskula-aws Nov 19, 2023
f35cc38
first draft tests to check coverage
jcjaskula-aws Nov 20, 2023
fa4cb3a
add more tests
jcjaskula-aws Nov 20, 2023
5ddf96e
add test with parametric global phase
jcjaskula-aws Nov 20, 2023
3c0d766
add test for neg control qubit printing
jcjaskula-aws Nov 20, 2023
9f8a547
clean up
jcjaskula-aws Nov 20, 2023
76989f7
simplify ctrl-gphase transform
jcjaskula-aws Nov 20, 2023
f1204d0
feat: add str, repr and getitem to BasisState
ajberdy Nov 21, 2023
f5d0511
add repr coverage
ajberdy Nov 21, 2023
f68b084
add index
jcjaskula-aws Nov 21, 2023
5c56724
add pop
jcjaskula-aws Nov 21, 2023
70844ee
Merge branch 'basis-state-slice' into jcjaskula-aws/add_u_gate
jcjaskula-aws Nov 21, 2023
dabb790
fix phase target qubit
jcjaskula-aws Nov 21, 2023
31a60f8
fix typing
jcjaskula-aws Nov 21, 2023
0b02f14
add index and pop tests
jcjaskula-aws Nov 21, 2023
0a7985e
fix code coverage
jcjaskula-aws Nov 21, 2023
c75f331
move unitary matrices
jcjaskula-aws Nov 21, 2023
df1b1a9
use a subindex in MomentKey
jcjaskula-aws Nov 23, 2023
daa9cd9
print global phase integration
jcjaskula-aws Nov 24, 2023
9e5f7dc
fix docstrings
jcjaskula-aws Nov 24, 2023
aa46064
fix circuits zero total global phase
jcjaskula-aws Nov 24, 2023
590d9c8
fix edge cases
jcjaskula-aws Nov 24, 2023
dec59aa
fix to_unitary
jcjaskula-aws Nov 24, 2023
ef91d3d
temporary fix that checks classname
jcjaskula-aws Nov 25, 2023
dfdecfe
clean up test conditions
jcjaskula-aws Nov 26, 2023
e5eaa21
change logic according to feedback
jcjaskula-aws Nov 28, 2023
b8f0f33
update docstring
jcjaskula-aws Nov 28, 2023
244d708
clean tests
jcjaskula-aws Nov 28, 2023
c185e41
update tests
jcjaskula-aws Nov 28, 2023
4efb8bc
replace control symbols
jcjaskula-aws Nov 30, 2023
ccb81fa
use box drawing characters
jcjaskula-aws Nov 30, 2023
4aa7545
Revert "use box drawing characters"
jcjaskula-aws Dec 1, 2023
c2291f8
Revert "replace control symbols"
jcjaskula-aws Dec 1, 2023
9386bca
simplify all gphase case
jcjaskula-aws Dec 1, 2023
ecbdff1
change preprare_y_axis function name
jcjaskula-aws Dec 3, 2023
696974d
Merge branch 'main' into jcjaskula-aws/add_u_gate
jcjaskula-aws Dec 11, 2023
ad1053f
create an helper function to compute the global phase
jcjaskula-aws Dec 11, 2023
cb6baac
make control_basis_state more explicit
jcjaskula-aws Dec 11, 2023
52ce20d
add comment and clean grouping
jcjaskula-aws Dec 11, 2023
d61a0d1
add test_only_neg_control_qubits
jcjaskula-aws Dec 11, 2023
b14c03b
parametrize test_one_gate_with_global_phase
jcjaskula-aws Dec 12, 2023
095fb68
reformat
jcjaskula-aws Dec 12, 2023
8ebb0ce
change to printing with fixed precision
jcjaskula-aws Dec 12, 2023
35a92cd
Merge branch 'main' into jcjaskula-aws/add_u_gate
jcjaskula-aws Dec 18, 2023
0626370
Merge branch 'main' into jcjaskula-aws/add_u_gate
jcjaskula-aws Dec 18, 2023
304a01d
Merge branch 'main' into jcjaskula-aws/add_u_gate
jcjaskula-aws Dec 20, 2023
b6e9563
fix docstring
jcjaskula-aws Dec 20, 2023
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
11 changes: 9 additions & 2 deletions src/braket/circuits/braket_program_context.py
Original file line number Diff line number Diff line change
Expand Up @@ -56,8 +56,15 @@
user_defined_gate = self.is_user_defined_gate(name)
return name in BRAKET_GATES and not user_defined_gate

def add_phase_instruction(self, target: tuple[int], phase_value: int) -> None:
raise NotImplementedError
def add_phase_instruction(self, target: tuple[int], phase_value: float) -> None:
"""Add a global phase to the circuit.

Args:
target (tuple[int]): Unused
phase_value (float): The phase value to be applied
"""
instruction = Instruction(BRAKET_GATES["gphase"](phase_value))
self._circuit.add_instruction(instruction)

Check warning on line 67 in src/braket/circuits/braket_program_context.py

View check run for this annotation

Codecov / codecov/patch

src/braket/circuits/braket_program_context.py#L66-L67

Added lines #L66 - L67 were not covered by tests

def add_gate_instruction(
self, gate_name: str, target: tuple[int], *params, ctrl_modifiers: list[int], power: float
Expand Down
2 changes: 1 addition & 1 deletion src/braket/circuits/gate.py
Original file line number Diff line number Diff line change
Expand Up @@ -198,7 +198,7 @@ def _to_openqasm(

return (
f"{inv_prefix}{power_prefix}{control_prefix}"
f"{self._qasm_name}{param_string} {', '.join(qubits)};"
f"{self._qasm_name}{param_string}{','.join([f' {qubit}' for qubit in qubits])};"
jcjaskula-aws marked this conversation as resolved.
Show resolved Hide resolved
)

@property
Expand Down
192 changes: 192 additions & 0 deletions src/braket/circuits/gates.py
Original file line number Diff line number Diff line change
Expand Up @@ -189,6 +189,80 @@
Gate.register_gate(I)


class GPhase(AngledGate):
"""Z-axis rotation gate.
jcjaskula-aws marked this conversation as resolved.
Show resolved Hide resolved

jcjaskula-aws marked this conversation as resolved.
Show resolved Hide resolved
Args:
angle (Union[FreeParameterExpression, float]): angle in radians.
"""

def __init__(self, angle: Union[FreeParameterExpression, float]):
# Avoid parent constructor because _qubit_count must be zero
self._qubit_count = self.fixed_qubit_count()
self._ascii_symbols = []

if angle is None:
raise ValueError("angle must not be None")

Check warning on line 205 in src/braket/circuits/gates.py

View check run for this annotation

Codecov / codecov/patch

src/braket/circuits/gates.py#L205

Added line #L205 was not covered by tests
jcjaskula-aws marked this conversation as resolved.
Show resolved Hide resolved
if isinstance(angle, FreeParameterExpression):
self._parameters = [angle]
else:
self._parameters = [float(angle)] # explicit casting in case angle is e.g. np.float32

@property
def _qasm_name(self) -> str:
return "gphase"

Check warning on line 213 in src/braket/circuits/gates.py

View check run for this annotation

Codecov / codecov/patch

src/braket/circuits/gates.py#L213

Added line #L213 was not covered by tests

def adjoint(self) -> list[Gate]:
return [GPhase(-self.angle)]

def to_matrix(self) -> np.ndarray:
return np.exp(1j * self.angle) * np.eye(1)
jcjaskula-aws marked this conversation as resolved.
Show resolved Hide resolved

def bind_values(self, **kwargs) -> AngledGate:
return get_angle(self, **kwargs)

@staticmethod
def fixed_qubit_count() -> int:
return 0

@staticmethod
@circuit.subroutine(register=True)
def gphase(
angle: Union[FreeParameterExpression, float],
*,
control: Optional[QubitSetInput] = None,
control_state: Optional[BasisStateInput] = None,
power: float = 1,
) -> Instruction:
"""Registers this function into the circuit class.

jcjaskula-aws marked this conversation as resolved.
Show resolved Hide resolved
Args:
angle (Union[FreeParameterExpression, float]): Phase in radians.
control (Optional[QubitSetInput]): Control qubit(s). Default None.
control_state (Optional[BasisStateInput]): Quantum state on which to control the
operation. Must be a binary sequence of same length as number of qubits in
`control`. Will be ignored if `control` is not present. May be represented as a
string, list, or int. For example "0101", [0, 1, 0, 1], 5 all represent
controlling on qubits 0 and 2 being in the \\|0⟩ state and qubits 1 and 3 being
in the \\|1⟩ state. Default "1" * len(control).
power (float): Integer or fractional power to raise the gate to. Negative
powers will be split into an inverse, accompanied by the positive power.
Default 1.

Returns:
Instruction: GPhase instruction d.
jcjaskula-aws marked this conversation as resolved.
Show resolved Hide resolved

Examples:
>>> circ = Circuit().gphase(0.45)
"""
return [
Instruction(GPhase(angle), control=control, control_state=control_state, power=power)
]
jcjaskula-aws marked this conversation as resolved.
Show resolved Hide resolved


Gate.register_gate(GPhase)


class X(Gate):
"""Pauli-X gate."""

Expand Down Expand Up @@ -1091,6 +1165,124 @@
Gate.register_gate(PhaseShift)


class U(TripleAngledGate):
"""Parameterized primitive gate for OpenQASM simulator
jcjaskula-aws marked this conversation as resolved.
Show resolved Hide resolved

jcjaskula-aws marked this conversation as resolved.
Show resolved Hide resolved
Args:
angle_1 (Union[FreeParameterExpression, float]): theta angle in radians.
angle_2 (Union[FreeParameterExpression, float]): phi angle in radians.
angle_3 (Union[FreeParameterExpression, float]): lambda angle in radians.
"""

def __init__(
self,
angle_1: Union[FreeParameterExpression, float],
angle_2: Union[FreeParameterExpression, float],
angle_3: Union[FreeParameterExpression, float],
Comment on lines +1420 to +1422
Copy link
Contributor

Choose a reason for hiding this comment

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

Can we add defaults for this gate as we did with MS? That way we can users can define a u2 gate by only providing relevant params.

Copy link
Contributor

Choose a reason for hiding this comment

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

Might be tricky since only the first gate would have a default, so maybe we circle back to this in another pr

Copy link
Contributor

Choose a reason for hiding this comment

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

Maybe we just add documentation that setting angle_1 to pi/2 is equivalent to U2

Copy link
Contributor Author

@jcjaskula-aws jcjaskula-aws Dec 11, 2023

Choose a reason for hiding this comment

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

Looks like u2/u3 are defined with a combination of gphase and U here. I like the idea to define u_i / update the docstring in another PR.

):
super().__init__(
angle_1=angle_1,
angle_2=angle_2,
angle_3=angle_3,
qubit_count=None,
ascii_symbols=[_multi_angled_ascii_characters("U", angle_1, angle_2, angle_3)],
)

@property
def _qasm_name(self) -> str:
return "U"

def to_matrix(self) -> np.ndarray:
r"""
Generate parameterized Unitary matrix.
https://openqasm.com/language/gates.html#built-in-gates

jcjaskula-aws marked this conversation as resolved.
Show resolved Hide resolved
Unitary matrix:
jcjaskula-aws marked this conversation as resolved.
Show resolved Hide resolved
.. math:: \mathtt{U}(\theta, \phi, \lambda) = \begin{bmatrix}
\cos{(\theta/2)} & -e^{i \lambda} \sin{(\theta/2)} \\
e^{i \phi} \sin{(\theta/2)} & -e^{i (\phi + \lambda)} \cos{(\theta/2)}
\end{bmatrix}.

Returns:
np.ndarray: U Matrix
"""
_theta = self.angle_1
_phi = self.angle_2
_lambda = self.angle_3
return np.array(
[
[
np.cos(_theta / 2),
-np.exp(1j * _lambda) * np.sin(_theta / 2),
],
[
np.exp(1j * _phi) * np.sin(_theta / 2),
np.exp(1j * (_phi + _lambda)) * np.cos(_theta / 2),
],
]
)

def adjoint(self) -> list[Gate]:
return [U(-self.angle_1, -self.angle_3, -self.angle_2)]

@staticmethod
def fixed_qubit_count() -> int:
return 1

def bind_values(self, **kwargs) -> TripleAngledGate:
return _get_angles(self, **kwargs)

@staticmethod
@circuit.subroutine(register=True)
def u(
target: QubitSetInput,
angle_1: Union[FreeParameterExpression, float],
angle_2: Union[FreeParameterExpression, float],
angle_3: Union[FreeParameterExpression, float],
*,
control: Optional[QubitSetInput] = None,
control_state: Optional[BasisStateInput] = None,
power: float = 1,
) -> Iterable[Instruction]:
"""Registers this function into the circuit class.

jcjaskula-aws marked this conversation as resolved.
Show resolved Hide resolved
Args:
target (QubitSetInput): Target qubit(s)
angle_1 (Union[FreeParameterExpression, float]): theta angle in radians.
angle_2 (Union[FreeParameterExpression, float]): phi angle in radians.
angle_3 (Union[FreeParameterExpression, float]): lambda angle in radians.
control (Optional[QubitSetInput]): Control qubit(s). Default None.
control_state (Optional[BasisStateInput]): Quantum state on which to control the
operation. Must be a binary sequence of same length as number of qubits in
`control`. Will be ignored if `control` is not present. May be represented as a
string, list, or int. For example "0101", [0, 1, 0, 1], 5 all represent
controlling on qubits 0 and 2 being in the \\|0⟩ state and qubits 1 and 3 being
in the \\|1⟩ state. Default "1" * len(control).
power (float): Integer or fractional power to raise the gate to. Negative
powers will be split into an inverse, accompanied by the positive power.
Default 1.

Returns:
Iterable[Instruction]: U instruction.

Examples:
>>> circ = Circuit().u(0, 0.15, 0.34, 0.52)
"""
return [
Instruction(
U(angle_1, angle_2, angle_3),
target=qubit,
control=control,
control_state=control_state,
power=power,
)
for qubit in QubitSet(target)
]


Gate.register_gate(U)


# Two qubit gates #


Expand Down
2 changes: 2 additions & 0 deletions src/braket/circuits/translations.py
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@
from braket.ir.jaqcd.program_v1 import Results

BRAKET_GATES = {
"gphase": braket_gates.GPhase,
"i": braket_gates.I,
"h": braket_gates.H,
"x": braket_gates.X,
Expand All @@ -55,6 +56,7 @@
"rx": braket_gates.Rx,
"ry": braket_gates.Ry,
"rz": braket_gates.Rz,
"U": braket_gates.U,
"swap": braket_gates.Swap,
"iswap": braket_gates.ISwap,
"pswap": braket_gates.PSwap,
Expand Down
35 changes: 32 additions & 3 deletions test/unit_tests/braket/circuits/test_gates.py
Original file line number Diff line number Diff line change
Expand Up @@ -35,12 +35,17 @@
from braket.pulse import ArbitraryWaveform, Frame, Port, PulseSequence


class NoTarget:
pass


class TripleAngle:
pass


testdata = [
(Gate.H, "h", ir.H, [SingleTarget], {}),
(Gate.GPhase, "gphase", None, [NoTarget, Angle], {}),
(Gate.I, "i", ir.I, [SingleTarget], {}),
(Gate.X, "x", ir.X, [SingleTarget], {}),
(Gate.Y, "y", ir.Y, [SingleTarget], {}),
Expand All @@ -54,6 +59,7 @@ class TripleAngle:
(Gate.Rx, "rx", ir.Rx, [SingleTarget, Angle], {}),
(Gate.Ry, "ry", ir.Ry, [SingleTarget, Angle], {}),
(Gate.Rz, "rz", ir.Rz, [SingleTarget, Angle], {}),
(Gate.U, "u", None, [SingleTarget, TripleAngle], {}),
jcjaskula-aws marked this conversation as resolved.
Show resolved Hide resolved
(Gate.CNot, "cnot", ir.CNot, [SingleTarget, SingleControl], {}),
(Gate.CV, "cv", ir.CV, [SingleTarget, SingleControl], {}),
(Gate.CCNot, "ccnot", ir.CCNot, [SingleTarget, DoubleControl], {}),
Expand Down Expand Up @@ -118,9 +124,11 @@ class TripleAngle:
]

parameterizable_gates = [
Gate.GPhase,
Gate.Rx,
Gate.Ry,
Gate.Rz,
Gate.U,
Gate.PhaseShift,
Gate.PSwap,
Gate.XX,
Expand All @@ -147,6 +155,10 @@ class TripleAngle:
]


def no_target_valid_input(**kwargs):
return {}


def single_target_valid_input(**kwargs):
return {"target": 2}

Expand Down Expand Up @@ -193,6 +205,7 @@ def two_dimensional_matrix_valid_input(**kwargs):


valid_ir_switcher = {
"NoTarget": no_target_valid_input,
"SingleTarget": single_target_valid_input,
"DoubleTarget": double_target_valid_ir_input,
"Angle": angle_valid_input,
Expand Down Expand Up @@ -234,7 +247,9 @@ def create_valid_target_input(irsubclasses):
qubit_set = []
# based on the concept that control goes first in target input
for subclass in irsubclasses:
if subclass == SingleTarget:
if subclass == NoTarget:
qubit_set.extend(list(no_target_valid_input().values()))
elif subclass == SingleTarget:
qubit_set.extend(list(single_target_valid_input().values()))
elif subclass == DoubleTarget:
qubit_set.extend(list(double_target_valid_ir_input().values()))
Expand Down Expand Up @@ -282,7 +297,7 @@ def calculate_qubit_count(irsubclasses):
qubit_count += 2
elif subclass == MultiTarget:
qubit_count += 3
elif subclass in (Angle, TwoDimensionalMatrix, TripleAngle):
elif subclass in (NoTarget, Angle, TwoDimensionalMatrix, TripleAngle):
pass
else:
raise ValueError("Invalid subclass")
Expand Down Expand Up @@ -434,6 +449,18 @@ def test_ir_gate_level(testclass, subroutine_name, irclass, irsubclasses, kwargs
OpenQASMSerializationProperties(qubit_reference_type=QubitReferenceType.PHYSICAL),
"rz(0.17) $4;",
),
(
Gate.U(angle_1=0.17, angle_2=3.45, angle_3=5.21),
[4],
OpenQASMSerializationProperties(qubit_reference_type=QubitReferenceType.VIRTUAL),
"U(0.17, 3.45, 5.21) q[4];",
),
(
Gate.U(angle_1=0.17, angle_2=3.45, angle_3=5.21),
[4],
OpenQASMSerializationProperties(qubit_reference_type=QubitReferenceType.PHYSICAL),
"U(0.17, 3.45, 5.21) $4;",
),
(
Gate.XX(angle=0.17),
[4, 5],
Expand Down Expand Up @@ -859,6 +886,8 @@ def test_gate_subroutine(testclass, subroutine_name, irclass, irsubclasses, kwar
subroutine_input = {"target": multi_targets}
if Angle in irsubclasses:
subroutine_input.update(angle_valid_input())
if TripleAngle in irsubclasses:
subroutine_input.update(triple_angle_valid_input())
assert subroutine(**subroutine_input) == Circuit(instruction_list)


Expand Down Expand Up @@ -925,7 +954,7 @@ def test_large_unitary():

@pytest.mark.parametrize("gate", parameterizable_gates)
def test_bind_values(gate):
triple_angled = gate.__name__ in ("MS",)
triple_angled = gate.__name__ in ("MS", "U")
num_params = 3 if triple_angled else 1
thetas = [FreeParameter(f"theta_{i}") for i in range(num_params)]
mapping = {f"theta_{i}": i for i in range(num_params)}
Expand Down