From d3cb14e05ed54b7137e207c181e4639f498f31f5 Mon Sep 17 00:00:00 2001 From: Ryan Shaffer <3620100+rmshaffer@users.noreply.github.com> Date: Mon, 13 May 2024 15:55:50 -0400 Subject: [PATCH 01/17] Add simulator module with mcm simulator implementation --- setup.py | 7 +- src/autoqasm/simulator/__init__.py | 24 ++ src/autoqasm/simulator/conversion.py | 52 +++++ src/autoqasm/simulator/linalg_utils.py | 93 ++++++++ src/autoqasm/simulator/native_interpreter.py | 151 ++++++++++++ src/autoqasm/simulator/program_context.py | 214 ++++++++++++++++++ src/autoqasm/simulator/simulation.py | 65 ++++++ src/autoqasm/simulator/simulator.py | 114 ++++++++++ test/unit_tests/autoqasm/test_api.py | 4 +- .../autoqasm/test_native_interpreter.py | 42 ++++ test/unit_tests/autoqasm/test_parameters.py | 4 +- tox.ini | 4 +- 12 files changed, 763 insertions(+), 11 deletions(-) create mode 100644 src/autoqasm/simulator/__init__.py create mode 100644 src/autoqasm/simulator/conversion.py create mode 100644 src/autoqasm/simulator/linalg_utils.py create mode 100644 src/autoqasm/simulator/native_interpreter.py create mode 100644 src/autoqasm/simulator/program_context.py create mode 100644 src/autoqasm/simulator/simulation.py create mode 100644 src/autoqasm/simulator/simulator.py create mode 100644 test/unit_tests/autoqasm/test_native_interpreter.py diff --git a/setup.py b/setup.py index a06cd03..677d153 100644 --- a/setup.py +++ b/setup.py @@ -25,11 +25,8 @@ package_dir={"": "src"}, install_requires=[ # Pin the latest commit of mcm-sim branch of amazon-braket/amazon-braket-sdk-python.git - # and amazon-braket/amazon-braket-default-simulator-python.git to get the version of the - # simulator that supports the mcm=True argument for Monte Carlo simulation of mid-circuit - # measurement, which AutoQASM requires. - "amazon-braket-sdk @ git+https://github.com/amazon-braket/amazon-braket-sdk-python.git@ff73de68cf6ac2d0a921e8fe62693e5b9ae2e321#egg=amazon-braket-sdk", # noqa E501 - "amazon-braket-default-simulator @ git+https://github.com/amazon-braket/amazon-braket-default-simulator-python.git@ab068c860963c29842d7649c741f88da669597eb#egg=amazon-braket-default-simulator", # noqa E501 + "amazon-braket-sdk @ git+https://github.com/amazon-braket/amazon-braket-sdk-python.git@fe601096b7bba2a1bd8700896080d93672732c4c#egg=amazon-braket-sdk", # noqa E501 + "amazon-braket-default-simulator>=1.23.2", "oqpy~=0.3.5", "diastatic-malt", "numpy", diff --git a/src/autoqasm/simulator/__init__.py b/src/autoqasm/simulator/__init__.py new file mode 100644 index 0000000..b95509e --- /dev/null +++ b/src/autoqasm/simulator/__init__.py @@ -0,0 +1,24 @@ +# Copyright Amazon.com Inc. or its affiliates. All Rights Reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"). You +# may not use this file except in compliance with the License. A copy of +# the License is located at +# +# http://aws.amazon.com/apache2.0/ +# +# or in the "license" file accompanying this file. This file is +# distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF +# ANY KIND, either express or implied. See the License for the specific +# language governing permissions and limitations under the License. + +""" +TODO: Module description + +TODO: Example usage + +.. code-block:: python + + # Python code here +""" + +from .simulator import McmSimulator # noqa: F401 diff --git a/src/autoqasm/simulator/conversion.py b/src/autoqasm/simulator/conversion.py new file mode 100644 index 0000000..acf92e3 --- /dev/null +++ b/src/autoqasm/simulator/conversion.py @@ -0,0 +1,52 @@ +# Copyright Amazon.com Inc. or its affiliates. All Rights Reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"). You +# may not use this file except in compliance with the License. A copy of +# the License is located at +# +# http://aws.amazon.com/apache2.0/ +# +# or in the "license" file accompanying this file. This file is +# distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF +# ANY KIND, either express or implied. See the License for the specific +# language governing permissions and limitations under the License. + +from functools import singledispatch +from typing import Any, Union + +import numpy as np +from braket.default_simulator.openqasm._helpers.casting import convert_bool_array_to_string +from braket.default_simulator.openqasm.parser.openqasm_ast import ( + ArrayLiteral, + BitstringLiteral, + BooleanLiteral, + FloatLiteral, + IntegerLiteral, +) + +LiteralType = Union[BooleanLiteral, IntegerLiteral, FloatLiteral, ArrayLiteral, BitstringLiteral] + + +@singledispatch +def convert_to_output(value: LiteralType) -> Any: + raise TypeError(f"converting {value} to output") + + +@convert_to_output.register(IntegerLiteral) +@convert_to_output.register(FloatLiteral) +@convert_to_output.register(BooleanLiteral) +@convert_to_output.register(BitstringLiteral) +def _(value): + return value.value + + +@convert_to_output.register(BitstringLiteral) +def _(value): + return np.array(np.binary_repr(value.value, value.width)) + + +@convert_to_output.register +def _(value: ArrayLiteral): + if isinstance(value.values[0], BooleanLiteral): + return convert_bool_array_to_string(value) + return np.array([convert_to_output(x) for x in value.values]) diff --git a/src/autoqasm/simulator/linalg_utils.py b/src/autoqasm/simulator/linalg_utils.py new file mode 100644 index 0000000..0b3d50c --- /dev/null +++ b/src/autoqasm/simulator/linalg_utils.py @@ -0,0 +1,93 @@ +# Copyright Amazon.com Inc. or its affiliates. All Rights Reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"). You +# may not use this file except in compliance with the License. A copy of +# the License is located at +# +# http://aws.amazon.com/apache2.0/ +# +# or in the "license" file accompanying this file. This file is +# distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF +# ANY KIND, either express or implied. See the License for the specific +# language governing permissions and limitations under the License. + +import itertools +from collections.abc import Iterable + +import numpy as np + + +def measurement_sample(prob: float, target_count: int) -> tuple[int]: + """_summary_ + + Args: + prob (float): _description_ + target_count (int): _description_ + + Returns: + tuple[int]: _description_ + """ + basis_states = np.array(list(itertools.product([0, 1], repeat=target_count))) + outcome_idx = np.random.choice(list(range(2**target_count)), p=prob) + return tuple(basis_states[outcome_idx]) + + +def measurement_collapse_dm( + dm_tensor: np.ndarray, targets: Iterable[int], outcomes: np.ndarray +) -> np.ndarray: + """_summary_ + + Args: + dm_tensor (np.ndarray): _description_ + targets (Iterable[int]): _description_ + outcomes (np.ndarray): _description_ + + Returns: + np.ndarray: _description_ + """ + # TODO: This needs to be modified to not delete qubits + + # move the target qubit to the front of axes + qubit_count = int(np.log2(dm_tensor.shape[0])) + unused_idxs = [idx for idx in range(qubit_count) if idx not in targets] + unused_idxs = [ + p + i * qubit_count for i in range(2) for p in unused_idxs + ] # convert indices to dm form + target_indx = [ + p + i * qubit_count for i in range(2) for p in targets + ] # convert indices to dm form + permutation = target_indx + unused_idxs + inverse_permutation = np.argsort(permutation) + + # collapse the density matrix based on measuremnt outcome + outcomes = tuple(i for _ in range(2) for i in outcomes) + new_dm_tensor = np.zeros_like(dm_tensor) + new_dm_tensor[outcomes] = np.transpose(dm_tensor, permutation)[outcomes] + new_dm_tensor = np.transpose(new_dm_tensor, inverse_permutation) + + # normalize + new_trace = np.trace(np.reshape(new_dm_tensor, (2**qubit_count, 2**qubit_count))) + new_dm_tensor = new_dm_tensor / new_trace + return new_dm_tensor + + +def measurement_collapse_sv( + state_vector: np.ndarray, targets: Iterable[int], outcome: np.ndarray +) -> np.ndarray: + """_summary_ + + Args: + state_vector (np.ndarray): _description_ + targets (Iterable[int]): _description_ + outcome (np.ndarray): _description_ + + Returns: + np.ndarray: _description_ + """ + qubit_count = int(np.log2(state_vector.size)) + state_tensor = state_vector.reshape([2] * qubit_count) + for qubit, measurement in zip(targets, outcome): + state_tensor[(slice(None),) * qubit + (int(not measurement),)] = 0 + + state_tensor /= np.linalg.norm(state_tensor) + return state_tensor.flatten() diff --git a/src/autoqasm/simulator/native_interpreter.py b/src/autoqasm/simulator/native_interpreter.py new file mode 100644 index 0000000..9367f06 --- /dev/null +++ b/src/autoqasm/simulator/native_interpreter.py @@ -0,0 +1,151 @@ +# Copyright Amazon.com Inc. or its affiliates. All Rights Reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"). You +# may not use this file except in compliance with the License. A copy of +# the License is located at +# +# http://aws.amazon.com/apache2.0/ +# +# or in the "license" file accompanying this file. This file is +# distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF +# ANY KIND, either express or implied. See the License for the specific +# language governing permissions and limitations under the License. + +from copy import deepcopy +from functools import singledispatchmethod +from logging import Logger +from typing import Any, List, Optional, Union + +from braket.default_simulator.openqasm._helpers.casting import cast_to, wrap_value_into_literal +from braket.default_simulator.openqasm.interpreter import Interpreter +from braket.default_simulator.openqasm.parser.openqasm_ast import ( + ArrayLiteral, + BitType, + BooleanLiteral, + ClassicalDeclaration, + IndexedIdentifier, + IODeclaration, + IOKeyword, + QASMNode, + QuantumMeasurement, + QuantumMeasurementStatement, + QuantumReset, + QubitDeclaration, +) +from braket.default_simulator.openqasm.parser.openqasm_parser import parse +from braket.default_simulator.simulation import Simulation +from openqasm3.ast import IntegerLiteral + +from autoqasm.simulator.program_context import McmProgramContext + + +class NativeInterpreter(Interpreter): + def __init__( + self, + simulation: Simulation, + context: Optional[McmProgramContext] = None, + logger: Optional[Logger] = None, + ): + self.simulation = simulation + context = context or McmProgramContext() + super().__init__(context, logger) + + def simulate( + self, + source: str, + inputs: Optional[dict[str, Any]] = None, + is_file: bool = False, + shots: int = 1, + ) -> dict[str, Any]: + """_summary_ + + Args: + source (str): _description_ + inputs (Optional[dict[str, Any]]): _description_. Defaults to None. + is_file (bool): _description_. Defaults to False. + shots (int): _description_. Defaults to 1. + + Returns: + dict[str, Any]: _description_ + """ + if inputs: + self.context.load_inputs(inputs) + + if is_file: + with open(source, encoding="utf-8", mode="r") as f: + source = f.read() + + program = parse(source) + for _ in range(shots): + program_copy = deepcopy(program) + self.visit(program_copy) + self.context.save_output_values() + self.context.num_qubits = 0 + self.simulation.reset() + return self.context.outputs + + @singledispatchmethod + def visit(self, node: Union[QASMNode, List[QASMNode]]) -> Optional[QASMNode]: + """Generic visit function for an AST node""" + return super().visit(node) + + @visit.register + def _(self, node: QubitDeclaration) -> None: + self.logger.debug(f"Qubit declaration: {node}") + size = self.visit(node.size).value if node.size else 1 + self.context.add_qubits(node.qubit.name, size) + self.simulation.add_qubits(size) + + @visit.register + def _(self, node: QuantumMeasurement) -> Union[BooleanLiteral, ArrayLiteral]: + self.logger.debug(f"Quantum measurement: {node}") + self.simulation.evolve(self.context.pop_instructions()) + targets = self.context.get_qubits(self.visit(node.qubit)) + outcome = self.simulation.measure(targets) + if len(targets) > 1 or ( + isinstance(node.qubit, IndexedIdentifier) + and not len(node.qubit.indices[0]) == 1 + and isinstance(node.qubit.indices[0], IntegerLiteral) + ): + return ArrayLiteral([BooleanLiteral(x) for x in outcome]) + return BooleanLiteral(outcome[0]) + + @visit.register + def _(self, node: QuantumMeasurementStatement) -> Union[BooleanLiteral, ArrayLiteral]: + self.logger.debug(f"Quantum measurement statement: {node}") + outcome = self.visit(node.measure) + current_value = self.context.get_value_by_identifier(node.target) + result_type = ( + BooleanLiteral + if isinstance(current_value, BooleanLiteral) or current_value is None + else BitType(size=IntegerLiteral(len(current_value.values))) + ) + value = cast_to(result_type, outcome) + self.context.update_value(node.target, value) + + @visit.register + def _(self, node: QuantumReset) -> None: + self.logger.debug(f"Quantum reset: {node}") + self.simulation.evolve(self.context.pop_instructions()) + targets = self.context.get_qubits(self.visit(node.qubits)) + outcome = self.simulation.measure(targets) + for qubit, result in zip(targets, outcome): + if result: + self.simulation.flip(qubit) + + @visit.register + def _(self, node: IODeclaration) -> None: + self.logger.debug(f"IO Declaration: {node}") + if node.io_identifier == IOKeyword.output: + if node.identifier.name not in self.context.outputs: + self.context.add_output(node.identifier.name) + self.context.declare_variable( + node.identifier.name, + node.type, + ) + else: # IOKeyword.input: + if node.identifier.name not in self.context.inputs: + raise NameError(f"Missing input variable '{node.identifier.name}'.") + init_value = wrap_value_into_literal(self.context.inputs[node.identifier.name]) + declaration = ClassicalDeclaration(node.type, node.identifier, init_value) + self.visit(declaration) diff --git a/src/autoqasm/simulator/program_context.py b/src/autoqasm/simulator/program_context.py new file mode 100644 index 0000000..ec9fc8a --- /dev/null +++ b/src/autoqasm/simulator/program_context.py @@ -0,0 +1,214 @@ +# Copyright Amazon.com Inc. or its affiliates. All Rights Reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"). You +# may not use this file except in compliance with the License. A copy of +# the License is located at +# +# http://aws.amazon.com/apache2.0/ +# +# or in the "license" file accompanying this file. This file is +# distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF +# ANY KIND, either express or implied. See the License for the specific +# language governing permissions and limitations under the License. + +from __future__ import annotations + +from collections.abc import Sequence +from functools import singledispatchmethod +from typing import Optional, Union + +import numpy as np +from braket.default_simulator.openqasm._helpers.arrays import ( + convert_discrete_set_to_list, + convert_range_def_to_slice, +) +from braket.default_simulator.openqasm.circuit import Circuit +from braket.default_simulator.openqasm.parser.openqasm_ast import ( + ClassicalType, + DiscreteSet, + Identifier, + IndexedIdentifier, + IntegerLiteral, + RangeDefinition, + SymbolLiteral, +) +from braket.default_simulator.openqasm.program_context import ProgramContext, Table +from braket.default_simulator.operation import GateOperation + +from autoqasm.simulator.conversion import convert_to_output + + +class QubitTable(Table): + def __init__(self): + super().__init__("Qubits") + + def _validate_qubit_in_range(self, qubit: int, register_name: str) -> None: + if qubit >= len(self[register_name]): + raise IndexError( + f"qubit register index `{qubit}` out of range for qubit register " + f"of length {len(self[register_name])} `{register_name}`." + ) + + @singledispatchmethod + def get_by_identifier(self, identifier: Union[Identifier, IndexedIdentifier]) -> tuple[int]: + """Convenience method to get an element with a possibly indexed identifier. + + Args: + identifier (Union[Identifier, IndexedIdentifier]): _description_ + + Returns: + tuple[int]: _description_ + """ + if identifier.name.startswith("$"): + return (int(identifier.name[1:]),) + return self[identifier.name] + + @get_by_identifier.register + def _(self, identifier: IndexedIdentifier) -> tuple[int]: # noqa: C901 + """When identifier is an IndexedIdentifier, function returns a tuple + corresponding to the elements referenced by the indexed identifier. + + Args: + identifier (IndexedIdentifier): _description_ + + Raises: + IndexError: Qubit register index out of range for specified register. + + Returns: + tuple[int]: _description_ + """ + name = identifier.name.name + indices = self.get_qubit_indices(identifier) + primary_index = indices[0] + + if isinstance(primary_index, (IntegerLiteral, SymbolLiteral)): + if isinstance(primary_index, IntegerLiteral): + self._validate_qubit_in_range(primary_index.value) + target = (self[name][0] + primary_index.value,) + elif isinstance(primary_index, RangeDefinition): + target = tuple(np.array(self[name])[convert_range_def_to_slice(primary_index)]) + # Discrete set + else: + index_list = convert_discrete_set_to_list(primary_index) + for index in index_list: + if isinstance(index, int): + self._validate_qubit_in_range(index) + target = tuple([self[name][0] + index for index in index_list]) + + if len(indices) == 2: + # used for gate calls on registers, index will be IntegerLiteral + secondary_index = indices[1].value + target = (target[secondary_index],) + + # validate indices manually, since we use addition instead of indexing to + # accommodate symbolic indices + for q in target: + if isinstance(q, int) and (relative_index := q - self[name][0]) >= len(self[name]): + raise IndexError( + f"qubit register index `{relative_index}` out of range for qubit register " + f"of length {len(self[name])} `{name}`." + ) + return target + + @staticmethod + def get_qubit_indices( + identifier: IndexedIdentifier, + ) -> list[IntegerLiteral | RangeDefinition | DiscreteSet]: + """_summary_ + + Args: + identifier (IndexedIdentifier): _description_ + + Raises: + IndexError: Index consists of multiple dimensions. + + Returns: + list[IntegerLiteral | RangeDefinition | DiscreteSet]: _description_ + """ + primary_index = identifier.indices[0] + + if isinstance(primary_index, list): + if len(primary_index) != 1: + raise IndexError("Cannot index multiple dimensions for qubits.") + primary_index = primary_index[0] + + if len(identifier.indices) == 1: + return [primary_index] + elif len(identifier.indices) == 2: + # used for gate calls on registers, index will be IntegerLiteral + secondary_index = identifier.indices[1][0] + return [primary_index, secondary_index] + else: + raise IndexError("Cannot index multiple dimensions for qubits.") + + def _get_indices_length( + self, + indices: Sequence[IntegerLiteral | SymbolLiteral | RangeDefinition | DiscreteSet], + ) -> int: + last_index = indices[-1] + + if isinstance(last_index, (IntegerLiteral, SymbolLiteral)): + return 1 + elif isinstance(last_index, RangeDefinition): + buffer = np.sign(last_index.step.value) if last_index.step is not None else 1 + start = last_index.start.value if last_index.start is not None else 0 + stop = last_index.end.value + buffer + step = last_index.step.value if last_index.step is not None else 1 + return (stop - start) // step + elif isinstance(last_index, DiscreteSet): + return len(last_index.values) + + def get_qubit_size(self, identifier: Union[Identifier, IndexedIdentifier]) -> int: + """_summary_ + + Args: + identifier (Union[Identifier, IndexedIdentifier]): _description_ + + Returns: + int: _description_ + """ + if isinstance(identifier, IndexedIdentifier): + indices = self.get_qubit_indices(identifier) + return self._get_indices_length(indices) + return len(self.get_by_identifier(identifier)) + + +class McmProgramContext(ProgramContext): + def __init__(self, circuit: Optional[Circuit] = None): + """ + Args: + circuit (Optional[Circuit]): A partially-built circuit to continue building with this + context. Default: None. + """ + self.outputs = {} + self._circuit = circuit or Circuit() + super(ProgramContext, self).__init__() + + def pop_instructions(self) -> list[GateOperation]: + """_summary_ + + Returns: + list[GateOperation]: _description_ + """ + instructions = self.circuit.instructions + self.circuit.instructions = [] + return instructions + + def add_output(self, output_name: str) -> None: + """_summary_ + + Args: + output_name (str): _description_ + """ + self.outputs[output_name] = [] + + def save_output_values(self) -> None: + """_summary_""" + if not self.outputs: + self.outputs = { + v: [] + for v in self.symbol_table.current_scope + if isinstance(self.get_type(v), ClassicalType) + } + for output, shot_data in self.outputs.items(): + shot_data.append(convert_to_output(self.get_value(output))) diff --git a/src/autoqasm/simulator/simulation.py b/src/autoqasm/simulator/simulation.py new file mode 100644 index 0000000..25eba9b --- /dev/null +++ b/src/autoqasm/simulator/simulation.py @@ -0,0 +1,65 @@ +# Copyright Amazon.com Inc. or its affiliates. All Rights Reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"). You +# may not use this file except in compliance with the License. A copy of +# the License is located at +# +# http://aws.amazon.com/apache2.0/ +# +# or in the "license" file accompanying this file. This file is +# distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF +# ANY KIND, either express or implied. See the License for the specific +# language governing permissions and limitations under the License. + +import numpy as np +from braket.default_simulator import StateVectorSimulation +from braket.default_simulator.gate_operations import PauliX +from braket.default_simulator.linalg_utils import marginal_probability + +from autoqasm.simulator.linalg_utils import measurement_collapse_sv, measurement_sample + + +class Simulation(StateVectorSimulation): + def add_qubits(self, num_qubits: int) -> None: + """_summary_ + + Args: + num_qubits (int): _description_ + """ + expanded_dims = np.expand_dims(self.state_vector, -1) + expanded_qubits = np.append( + expanded_dims, np.zeros((expanded_dims.size, 2**num_qubits - 1)), axis=-1 + ) + self._state_vector = expanded_qubits.flatten() + self._qubit_count += num_qubits + + def measure(self, targets: tuple[int]) -> tuple[int]: + """_summary_ + + Args: + targets (tuple[int]): _description_ + + Returns: + tuple[int]: _description_ + """ + mprob = marginal_probability(self.probabilities, targets) + outcome = measurement_sample(mprob, len(targets)) + self._state_vector = measurement_collapse_sv( + self._state_vector, + targets, + outcome, + ) + return outcome + + def reset(self) -> None: + """_summary_""" + self._state_vector = np.array([1], dtype=complex) + self._qubit_count = 0 + + def flip(self, target: int) -> None: + """_summary_ + + Args: + target (int): _description_ + """ + self.evolve([PauliX([target])]) diff --git a/src/autoqasm/simulator/simulator.py b/src/autoqasm/simulator/simulator.py new file mode 100644 index 0000000..d9be8ab --- /dev/null +++ b/src/autoqasm/simulator/simulator.py @@ -0,0 +1,114 @@ +# Copyright Amazon.com Inc. or its affiliates. All Rights Reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"). You +# may not use this file except in compliance with the License. A copy of +# the License is located at +# +# http://aws.amazon.com/apache2.0/ +# +# or in the "license" file accompanying this file. This file is +# distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF +# ANY KIND, either express or implied. See the License for the specific +# language governing permissions and limitations under the License. + +from braket.default_simulator import StateVectorSimulator +from braket.default_simulator.openqasm.circuit import Circuit +from braket.ir.openqasm import Program as OpenQASMProgram +from braket.task_result import AdditionalMetadata, TaskMetadata +from braket.tasks import GateModelQuantumTaskResult + +from autoqasm.simulator.native_interpreter import NativeInterpreter +from autoqasm.simulator.program_context import McmProgramContext +from autoqasm.simulator.simulation import Simulation + + +class McmSimulator(StateVectorSimulator): + DEVICE_ID = "autoqasm_mcm" + + def initialize_simulation(self, **kwargs) -> Simulation: + """ + Initialize simulation with mid-circuit measurement (MCM) support. + + Args: + `**kwargs`: qubit_count, shots, batch_size + + Returns: + Simulation: Initialized simulation. + """ + qubit_count = kwargs.get("qubit_count") + shots = kwargs.get("shots") + batch_size = kwargs.get("batch_size") + return Simulation(qubit_count, shots, batch_size) + + def create_program_context(self) -> McmProgramContext: + return McmProgramContext() + + def run( + self, + openqasm_ir: OpenQASMProgram, + shots: int = 0, + *, + batch_size: int = 1, + ) -> GateModelQuantumTaskResult: + """Executes the program specified by the supplied `circuit_ir` on the simulator. + + Args: + openqasm_ir (OpenQASMProgram): ir representation of a program specifying the + instructions to execute. + shots (int): The number of times to run the circuit. + batch_size (int): The size of the circuit partitions to contract, + if applying multiple gates at a time is desired; see `StateVectorSimulation`. + Must be a positive integer. + Defaults to 1, which means gates are applied one at a time without any + optimized contraction. + Returns: + GateModelQuantumTaskResult: object that represents the result + + Raises: + ValueError: If result types are not specified in the IR or sample is specified + as a result type when shots=0. Or, if StateVector and Amplitude result types + are requested when shots>0. + """ + is_file = openqasm_ir.source.endswith(".qasm") + simulation = self.initialize_simulation(qubit_count=0, shots=shots, batch_size=batch_size) + interpreter = NativeInterpreter(simulation=simulation) + + context = interpreter.simulate( + source=openqasm_ir.source, + inputs=openqasm_ir.inputs, + is_file=is_file, + shots=shots, + ) + + return GateModelQuantumTaskResult( + task_metadata=TaskMetadata.construct(id="", shots=shots), + additional_metadata=AdditionalMetadata.construct(), + measurements=context, + ) + + def _validate_input_provided(self, circuit: Circuit) -> None: + """ + Validate that requested circuit has all input parameters provided. + + Args: + circuit (Circuit): IR for the simulator. + + Raises: + NameError: If any the specified input parameters are not provided + """ + for instruction in circuit.instructions: + possible_parameters = "_angle", "_angle_1", "_angle_2" + for parameter_name in possible_parameters: + param = getattr(instruction, parameter_name, None) + if param is not None: + try: + float(param) + except TypeError: + missing_input = param.free_symbols.pop() + raise NameError(f"Missing input variable '{missing_input}'.") + for qubit in instruction.targets: + try: + float(qubit) + except TypeError: + missing_input = qubit.free_symbols.pop() + raise NameError(f"Missing input variable '{missing_input}'.") diff --git a/test/unit_tests/autoqasm/test_api.py b/test/unit_tests/autoqasm/test_api.py index 546716d..3d7b331 100644 --- a/test/unit_tests/autoqasm/test_api.py +++ b/test/unit_tests/autoqasm/test_api.py @@ -17,17 +17,17 @@ """ import pytest -from braket.default_simulator import StateVectorSimulator from braket.devices.local_simulator import LocalSimulator from braket.tasks.local_quantum_task import LocalQuantumTask import autoqasm as aq from autoqasm import errors from autoqasm.instructions import cnot, h, measure, rx, x +from autoqasm.simulator import McmSimulator def _test_on_local_sim(program: aq.Program, inputs=None) -> None: - device = LocalSimulator(backend=StateVectorSimulator()) + device = LocalSimulator(backend=McmSimulator()) task = device.run(program, shots=10, inputs=inputs or {}) assert isinstance(task, LocalQuantumTask) assert isinstance(task.result().measurements, dict) diff --git a/test/unit_tests/autoqasm/test_native_interpreter.py b/test/unit_tests/autoqasm/test_native_interpreter.py new file mode 100644 index 0000000..9b56b76 --- /dev/null +++ b/test/unit_tests/autoqasm/test_native_interpreter.py @@ -0,0 +1,42 @@ +# Copyright Amazon.com Inc. or its affiliates. All Rights Reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"). You +# may not use this file except in compliance with the License. A copy of +# the License is located at +# +# http://aws.amazon.com/apache2.0/ +# +# or in the "license" file accompanying this file. This file is +# distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF +# ANY KIND, either express or implied. See the License for the specific +# language governing permissions and limitations under the License. + +import pytest + +from autoqasm.simulator.native_interpreter import NativeInterpreter +from autoqasm.simulator.simulation import Simulation + + +@pytest.mark.parametrize( + "reset_instructions", + ( + "for int q in [0:2 - 1] {\n reset __qubits__[q];\n}", + "array[int[32], 2] __arr__ = {0, 1};\nfor int q in __arr__ {\n reset __qubits__[q];\n}", + "reset __qubits__[0];\nreset __qubits__[1];", + "reset __qubits__;", + ), +) +def test_reset(reset_instructions): + qasm = f""" + OPENQASM 3.0; + qubit[2] __qubits__; + x __qubits__[0]; + x __qubits__[1]; + {reset_instructions} + bit[2] __bit_0__ = "00"; + __bit_0__[0] = measure __qubits__[0]; + __bit_0__[1] = measure __qubits__[1]; + """ + + result = NativeInterpreter(Simulation(0, 0, 1)).simulate(qasm) + assert result["__bit_0__"] == ["00"] diff --git a/test/unit_tests/autoqasm/test_parameters.py b/test/unit_tests/autoqasm/test_parameters.py index 14a15ad..9dc36a9 100644 --- a/test/unit_tests/autoqasm/test_parameters.py +++ b/test/unit_tests/autoqasm/test_parameters.py @@ -16,17 +16,17 @@ import numpy as np import pytest from braket.circuits import FreeParameter -from braket.default_simulator import StateVectorSimulator from braket.devices.local_simulator import LocalSimulator from braket.tasks.local_quantum_task import LocalQuantumTask import autoqasm as aq from autoqasm import pulse from autoqasm.instructions import cnot, cphaseshift, gpi, h, measure, ms, rx, rz, x +from autoqasm.simulator import McmSimulator def _test_parametric_on_local_sim(program: aq.Program, inputs: dict[str, float]) -> np.ndarray: - device = LocalSimulator(backend=StateVectorSimulator()) + device = LocalSimulator(backend=McmSimulator()) task = device.run(program, shots=100, inputs=inputs) assert isinstance(task, LocalQuantumTask) assert isinstance(task.result().measurements, dict) diff --git a/tox.ini b/tox.ini index 4f451ae..c4ca0c6 100644 --- a/tox.ini +++ b/tox.ini @@ -133,5 +133,5 @@ commands = [test-deps] deps = # If you need to test on a certain branch, add @ after .git - git+https://github.com/amazon-braket/amazon-braket-sdk-python.git@ff73de68cf6ac2d0a921e8fe62693e5b9ae2e321 # mcm-sim branch - git+https://github.com/amazon-braket/amazon-braket-default-simulator-python.git@ab068c860963c29842d7649c741f88da669597eb # mcm-sim branch + git+https://github.com/amazon-braket/amazon-braket-sdk-python.git@fe601096b7bba2a1bd8700896080d93672732c4c # mcm-sim branch + git+https://github.com/amazon-braket/amazon-braket-default-simulator-python.git From 7e4441dd6cc42c1ee01b52d86e1ee7a04fa15b40 Mon Sep 17 00:00:00 2001 From: Ryan Shaffer <3620100+rmshaffer@users.noreply.github.com> Date: Tue, 14 May 2024 13:56:50 -0400 Subject: [PATCH 02/17] Rename build_if_necessary param --- setup.py | 2 +- src/autoqasm/program/program.py | 12 ++++++------ test/unit_tests/autoqasm/test_api.py | 8 ++++---- tox.ini | 2 +- 4 files changed, 12 insertions(+), 12 deletions(-) diff --git a/setup.py b/setup.py index 677d153..e45b277 100644 --- a/setup.py +++ b/setup.py @@ -25,7 +25,7 @@ package_dir={"": "src"}, install_requires=[ # Pin the latest commit of mcm-sim branch of amazon-braket/amazon-braket-sdk-python.git - "amazon-braket-sdk @ git+https://github.com/amazon-braket/amazon-braket-sdk-python.git@fe601096b7bba2a1bd8700896080d93672732c4c#egg=amazon-braket-sdk", # noqa E501 + "amazon-braket-sdk @ git+https://github.com/amazon-braket/amazon-braket-sdk-python.git@da74ee713c200ac0803704bb0b4c87f9d0100682#egg=amazon-braket-sdk", # noqa E501 "amazon-braket-default-simulator>=1.23.2", "oqpy~=0.3.5", "diastatic-malt", diff --git a/src/autoqasm/program/program.py b/src/autoqasm/program/program.py index c97c82a..97809db 100644 --- a/src/autoqasm/program/program.py +++ b/src/autoqasm/program/program.py @@ -121,7 +121,7 @@ def build(self, device: Device | str | None = None) -> Program: def to_ir( self, ir_type: IRType = IRType.OPENQASM, - allow_implicit_build: bool = False, + build_if_necessary: bool = False, serialization_properties: SerializationProperties = OpenQASMSerializationProperties(), ) -> str: """Serializes the program into an intermediate representation. @@ -129,20 +129,20 @@ def to_ir( Args: ir_type (IRType): The IRType to use for converting the program to its IR representation. Defaults to IRType.OPENQASM. - allow_implicit_build (bool): Whether to allow the program to be implicitly + build_if_necessary (bool): Whether to allow the program to be implicitly built as a side effect of calling this function. Defaults to False. serialization_properties (SerializationProperties): IR serialization configuration. Default to OpenQASMSerializationProperties(). Raises: ValueError: Raised if the supplied `ir_type` is not supported. - RuntimeError: Raised if `allow_implicit_build` is False, since a MainProgram object + RuntimeError: Raised if `build_if_necessary` is False, since a MainProgram object has not yet been built. Returns: str: A representation of the program in the `ir_type` format. """ - if not allow_implicit_build: + if not build_if_necessary: raise RuntimeError( "The AutoQASM program cannot be serialized because it has not yet been built. " "To serialize the program, first call build() to obtain a built Program object, " @@ -227,7 +227,7 @@ def make_bound_program(self, param_values: dict[str, float], strict: bool = Fals def to_ir( self, ir_type: IRType = IRType.OPENQASM, - allow_implicit_build: bool = True, + build_if_necessary: bool = True, serialization_properties: SerializationProperties = OpenQASMSerializationProperties(), ) -> str: """Serializes the program into an intermediate representation. @@ -235,7 +235,7 @@ def to_ir( Args: ir_type (IRType): The IRType to use for converting the program to its IR representation. Defaults to IRType.OPENQASM. - allow_implicit_build (bool): Whether to allow the program to be implicitly + build_if_necessary (bool): Whether to allow the program to be implicitly built as a side effect of calling this function. Defaults to True. This parameter is ignored for the Program class, since the program has already been built. diff --git a/test/unit_tests/autoqasm/test_api.py b/test/unit_tests/autoqasm/test_api.py index 3d7b331..3f4a584 100644 --- a/test/unit_tests/autoqasm/test_api.py +++ b/test/unit_tests/autoqasm/test_api.py @@ -952,11 +952,11 @@ def empty_program() -> None: def test_to_ir_implicit_build(empty_program) -> None: """Test that to_ir works as expected with and without implicit build.""" expected = """OPENQASM 3.0;""" - assert empty_program.build().to_ir(allow_implicit_build=False) == expected - assert empty_program.build().to_ir(allow_implicit_build=True) == expected - assert empty_program.to_ir(allow_implicit_build=True) == expected + assert empty_program.build().to_ir(build_if_necessary=False) == expected + assert empty_program.build().to_ir(build_if_necessary=True) == expected + assert empty_program.to_ir(build_if_necessary=True) == expected with pytest.raises(RuntimeError): - empty_program.to_ir(allow_implicit_build=False) + empty_program.to_ir(build_if_necessary=False) def test_main_no_return(): diff --git a/tox.ini b/tox.ini index c4ca0c6..572d24e 100644 --- a/tox.ini +++ b/tox.ini @@ -133,5 +133,5 @@ commands = [test-deps] deps = # If you need to test on a certain branch, add @ after .git - git+https://github.com/amazon-braket/amazon-braket-sdk-python.git@fe601096b7bba2a1bd8700896080d93672732c4c # mcm-sim branch + git+https://github.com/amazon-braket/amazon-braket-sdk-python.git@da74ee713c200ac0803704bb0b4c87f9d0100682 # mcm-sim branch git+https://github.com/amazon-braket/amazon-braket-default-simulator-python.git From ffb1dac2d42af7039f78992ab8c90b94c0482309 Mon Sep 17 00:00:00 2001 From: Ryan Shaffer <3620100+rmshaffer@users.noreply.github.com> Date: Tue, 14 May 2024 14:16:20 -0400 Subject: [PATCH 03/17] Move simulator tests into separate folder --- .../autoqasm/{ => simulator}/test_native_interpreter.py | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename test/unit_tests/autoqasm/{ => simulator}/test_native_interpreter.py (100%) diff --git a/test/unit_tests/autoqasm/test_native_interpreter.py b/test/unit_tests/autoqasm/simulator/test_native_interpreter.py similarity index 100% rename from test/unit_tests/autoqasm/test_native_interpreter.py rename to test/unit_tests/autoqasm/simulator/test_native_interpreter.py From fd046439fac3071a4c2acc81bcd3d00a41242605 Mon Sep 17 00:00:00 2001 From: Ryan Shaffer <3620100+rmshaffer@users.noreply.github.com> Date: Tue, 14 May 2024 14:47:22 -0400 Subject: [PATCH 04/17] Tests for NativeInterpreter --- test/resources/inputs.qasm | 8 +++ .../simulator/test_native_interpreter.py | 64 ++++++++++++++++--- 2 files changed, 63 insertions(+), 9 deletions(-) create mode 100644 test/resources/inputs.qasm diff --git a/test/resources/inputs.qasm b/test/resources/inputs.qasm new file mode 100644 index 0000000..7d779f6 --- /dev/null +++ b/test/resources/inputs.qasm @@ -0,0 +1,8 @@ +OPENQASM 3.0; +input float theta; +output bit return_value; +qubit[1] __qubits__; +rx(theta) __qubits__[0]; +bit __bit_0__; +__bit_0__ = measure __qubits__[0]; +return_value = __bit_0__; diff --git a/test/unit_tests/autoqasm/simulator/test_native_interpreter.py b/test/unit_tests/autoqasm/simulator/test_native_interpreter.py index 9b56b76..e7dcf8e 100644 --- a/test/unit_tests/autoqasm/simulator/test_native_interpreter.py +++ b/test/unit_tests/autoqasm/simulator/test_native_interpreter.py @@ -16,6 +16,8 @@ from autoqasm.simulator.native_interpreter import NativeInterpreter from autoqasm.simulator.simulation import Simulation +INPUTS_QASM = "test/resources/inputs.qasm" + @pytest.mark.parametrize( "reset_instructions", @@ -28,15 +30,59 @@ ) def test_reset(reset_instructions): qasm = f""" - OPENQASM 3.0; - qubit[2] __qubits__; - x __qubits__[0]; - x __qubits__[1]; - {reset_instructions} - bit[2] __bit_0__ = "00"; - __bit_0__[0] = measure __qubits__[0]; - __bit_0__[1] = measure __qubits__[1]; + OPENQASM 3.0; + qubit[2] __qubits__; + x __qubits__[0]; + {reset_instructions} + bit[2] __bit_0__ = "00"; + __bit_0__[0] = measure __qubits__[0]; + __bit_0__[1] = measure __qubits__[1]; """ - result = NativeInterpreter(Simulation(0, 0, 1)).simulate(qasm) assert result["__bit_0__"] == ["00"] + + +def test_inputs_outputs(): + with open(INPUTS_QASM, encoding="utf-8", mode="r") as f: + qasm = f.read() + + result = NativeInterpreter(Simulation(1, 1, 1)).simulate(qasm, inputs={"theta": 0.0}) + assert result["return_value"] == [0] + + +def test_inputs_outputs_from_file(): + result = NativeInterpreter(Simulation(1, 1, 1)).simulate( + INPUTS_QASM, inputs={"theta": 0.0}, is_file=True + ) + assert result["return_value"] == [0] + + +def test_missing_input(): + qasm = """ + OPENQASM 3.0; + input float theta; + """ + with pytest.raises(NameError, match="Missing input variable"): + NativeInterpreter(Simulation(0, 0, 1)).simulate(qasm) + + +def test_repeated_output_declaration(): + qasm = """ + OPENQASM 3.0; + output bit return_value; + output bit return_value; + return_value = 0; + """ + result = NativeInterpreter(Simulation(0, 0, 1)).simulate(qasm) + assert result["return_value"] == [0] + + +def test_qubit_register(): + qasm = """ + OPENQASM 3.0; + qubit[2] __qubits__; + x __qubits__[1]; + bit[2] __bit_0__ = measure __qubits__; + """ + result = NativeInterpreter(Simulation(2, 1, 1)).simulate(qasm) + assert result["__bit_0__"] == ["01"] From 0dc6f6eb8a04b77a1158f6e78bd325e10014945c Mon Sep 17 00:00:00 2001 From: Ryan Shaffer <3620100+rmshaffer@users.noreply.github.com> Date: Tue, 14 May 2024 17:00:57 -0400 Subject: [PATCH 05/17] Register entry point for LocalSimulator --- README.md | 4 ++-- examples/1_Getting_started_with_AutoQASM.ipynb | 2 +- examples/2_Expressing_classical_control_flow.ipynb | 2 +- examples/3_1_Iterative_phase_estimation.ipynb | 2 +- examples/3_2_magic_state_distillation.ipynb | 6 +++--- setup.py | 5 +++++ src/autoqasm/simulator/simulator.py | 2 +- 7 files changed, 14 insertions(+), 9 deletions(-) diff --git a/README.md b/README.md index 0484351..617db7c 100644 --- a/README.md +++ b/README.md @@ -96,13 +96,13 @@ AutoQASM can support subroutines and complex control flow. You can use the Pytho and quantum runtime side-by-side. There are rough edges at the moment, but we're actively smoothing them out! -The Amazon Braket local simulator supports AutoQASM programs as input. +AutoQASM includes a simulator which can be accessed using the Amazon Braket local simulator interface. Let's simulate the `conditional_multi_bell_states` program: ``` from braket.devices.local_simulator import LocalSimulator -device = LocalSimulator() +device = LocalSimulator("autoqasm") task = device.run(conditional_multi_bell_states, shots=100) result = task.result() ``` diff --git a/examples/1_Getting_started_with_AutoQASM.ipynb b/examples/1_Getting_started_with_AutoQASM.ipynb index e33d5b7..962db45 100644 --- a/examples/1_Getting_started_with_AutoQASM.ipynb +++ b/examples/1_Getting_started_with_AutoQASM.ipynb @@ -108,7 +108,7 @@ } ], "source": [ - "device = LocalSimulator()\n", + "device = LocalSimulator(\"autoqasm\")\n", "result = device.run(bell_state, shots=100).result()\n", "counts = Counter(result.measurements[\"return_value\"])\n", "print(\"measurement counts: \", counts)" diff --git a/examples/2_Expressing_classical_control_flow.ipynb b/examples/2_Expressing_classical_control_flow.ipynb index d4c5694..5f2726c 100644 --- a/examples/2_Expressing_classical_control_flow.ipynb +++ b/examples/2_Expressing_classical_control_flow.ipynb @@ -167,7 +167,7 @@ } ], "source": [ - "device = LocalSimulator()\n", + "device = LocalSimulator(\"autoqasm\")\n", "result = device.run(conditioned_bell, shots=500).result()\n", "counts = Counter(result.measurements[\"return_value\"])\n", "print(\"measurement counts: \", counts)\n", diff --git a/examples/3_1_Iterative_phase_estimation.ipynb b/examples/3_1_Iterative_phase_estimation.ipynb index 57ea403..72a5b2b 100644 --- a/examples/3_1_Iterative_phase_estimation.ipynb +++ b/examples/3_1_Iterative_phase_estimation.ipynb @@ -158,7 +158,7 @@ } ], "source": [ - "device = LocalSimulator()\n", + "device = LocalSimulator(\"autoqasm\")\n", "result = device.run(ipe, shots=100).result()\n", "counts = Counter(result.measurements[\"b0\"])\n", "print(\"measurement counts: \", counts)\n", diff --git a/examples/3_2_magic_state_distillation.ipynb b/examples/3_2_magic_state_distillation.ipynb index 0612f3e..dfed335 100644 --- a/examples/3_2_magic_state_distillation.ipynb +++ b/examples/3_2_magic_state_distillation.ipynb @@ -175,7 +175,7 @@ ], "source": [ "# Get measurement result\n", - "result = LocalSimulator().run(gate_teleportation, shots=100).result()\n", + "result = LocalSimulator(\"autoqasm\").run(gate_teleportation, shots=100).result()\n", "counts = Counter(result.measurements[\"return_value\"])\n", "print(\"measurement counts: \", counts)\n", "\n", @@ -315,7 +315,7 @@ ], "source": [ "n_shots = 1000\n", - "result = LocalSimulator().run(distillation, shots=n_shots).result()\n", + "result = LocalSimulator(\"autoqasm\").run(distillation, shots=n_shots).result()\n", "counts = Counter(result.measurements[\"c\"])\n", "print(\"measurement counts: \", counts)" ] @@ -435,7 +435,7 @@ } ], "source": [ - "result = LocalSimulator().run(distillation_rus, shots=20).result()\n", + "result = LocalSimulator(\"autoqasm\").run(distillation_rus, shots=20).result()\n", "counts = Counter(result.measurements[\"c2\"])\n", "probs = {str(k): v / sum(counts.values()) for k, v in counts.items()}\n", "\n", diff --git a/setup.py b/setup.py index e45b277..0cafbff 100644 --- a/setup.py +++ b/setup.py @@ -57,6 +57,11 @@ "tox", ], }, + entry_points={ + "braket.simulators": [ + "autoqasm = autoqasm.simulator.simulator:McmSimulator", + ] + }, include_package_data=True, url="https://github.com/amazon-braket/autoqasm", author="Amazon Web Services", diff --git a/src/autoqasm/simulator/simulator.py b/src/autoqasm/simulator/simulator.py index d9be8ab..d823648 100644 --- a/src/autoqasm/simulator/simulator.py +++ b/src/autoqasm/simulator/simulator.py @@ -23,7 +23,7 @@ class McmSimulator(StateVectorSimulator): - DEVICE_ID = "autoqasm_mcm" + DEVICE_ID = "autoqasm" def initialize_simulation(self, **kwargs) -> Simulation: """ From 8fad137c466f3690b08eb7e4d06b8894214ee203 Mon Sep 17 00:00:00 2001 From: Ryan Shaffer <3620100+rmshaffer@users.noreply.github.com> Date: Wed, 15 May 2024 14:56:33 -0400 Subject: [PATCH 06/17] Improve code coverage for program_context.py --- src/autoqasm/simulator/program_context.py | 9 ++- .../simulator/test_native_interpreter.py | 56 +++++++++++++++++++ 2 files changed, 62 insertions(+), 3 deletions(-) diff --git a/src/autoqasm/simulator/program_context.py b/src/autoqasm/simulator/program_context.py index ec9fc8a..03f8f30 100644 --- a/src/autoqasm/simulator/program_context.py +++ b/src/autoqasm/simulator/program_context.py @@ -83,7 +83,7 @@ def _(self, identifier: IndexedIdentifier) -> tuple[int]: # noqa: C901 if isinstance(primary_index, (IntegerLiteral, SymbolLiteral)): if isinstance(primary_index, IntegerLiteral): - self._validate_qubit_in_range(primary_index.value) + self._validate_qubit_in_range(primary_index.value, name) target = (self[name][0] + primary_index.value,) elif isinstance(primary_index, RangeDefinition): target = tuple(np.array(self[name])[convert_range_def_to_slice(primary_index)]) @@ -92,7 +92,7 @@ def _(self, identifier: IndexedIdentifier) -> tuple[int]: # noqa: C901 index_list = convert_discrete_set_to_list(primary_index) for index in index_list: if isinstance(index, int): - self._validate_qubit_in_range(index) + self._validate_qubit_in_range(index, name) target = tuple([self[name][0] + index for index in index_list]) if len(indices) == 2: @@ -157,6 +157,8 @@ def _get_indices_length( return (stop - start) // step elif isinstance(last_index, DiscreteSet): return len(last_index.values) + else: + raise NotImplementedError def get_qubit_size(self, identifier: Union[Identifier, IndexedIdentifier]) -> int: """_summary_ @@ -180,9 +182,10 @@ def __init__(self, circuit: Optional[Circuit] = None): circuit (Optional[Circuit]): A partially-built circuit to continue building with this context. Default: None. """ + super(ProgramContext, self).__init__() + self.qubit_mapping = QubitTable() self.outputs = {} self._circuit = circuit or Circuit() - super(ProgramContext, self).__init__() def pop_instructions(self) -> list[GateOperation]: """_summary_ diff --git a/test/unit_tests/autoqasm/simulator/test_native_interpreter.py b/test/unit_tests/autoqasm/simulator/test_native_interpreter.py index e7dcf8e..41413cb 100644 --- a/test/unit_tests/autoqasm/simulator/test_native_interpreter.py +++ b/test/unit_tests/autoqasm/simulator/test_native_interpreter.py @@ -86,3 +86,59 @@ def test_qubit_register(): """ result = NativeInterpreter(Simulation(2, 1, 1)).simulate(qasm) assert result["__bit_0__"] == ["01"] + + +def test_qubit_index_out_of_range(): + qasm = """ + OPENQASM 3.0; + qubit[2] __qubits__; + x __qubits__[4]; + """ + with pytest.raises(IndexError, match="qubit register index `4` out of range"): + NativeInterpreter(Simulation(2, 1, 1)).simulate(qasm) + + +def test_qubit_index_out_of_range_symbolic(): + qasm = """ + OPENQASM 3.0; + qubit[2] __qubits__; + for int[32] i in [0:3] { + x __qubits__[i]; + } + """ + with pytest.raises(IndexError, match="qubit register index `2` out of range"): + NativeInterpreter(Simulation(2, 1, 1)).simulate(qasm) + + +def test_physical_qubit_identifier(): + qasm = """ + OPENQASM 3.0; + x $0; + bit __bit_0__ = measure $0; + """ + result = NativeInterpreter(Simulation(1, 1, 1)).simulate(qasm) + assert result["__bit_0__"] == [1] + + +def test_qubit_register_indexing(): + qasm = """ + OPENQASM 3.0; + qubit[2] __qubits__; + int a = 0; + x __qubits__[a+1]; + h __qubits__[0:1]; + h __qubits__[{a, 1}]; + bit[2] __bit_0__ = measure __qubits__; + """ + result = NativeInterpreter(Simulation(1, 1, 1)).simulate(qasm) + assert result["__bit_0__"] == ["01"] + + +def test_qubit_register_indexing_multi_dimensions(): + qasm = """ + OPENQASM 3.0; + qubit[2] __qubits__; + x __qubits__[0, 1]; + """ + with pytest.raises(IndexError, match="Cannot index multiple dimensions for qubits."): + NativeInterpreter(Simulation(2, 1, 1)).simulate(qasm) From edd9e0725e8d558be1519750fc3c4d747dd0fcef Mon Sep 17 00:00:00 2001 From: Ryan Shaffer <3620100+rmshaffer@users.noreply.github.com> Date: Wed, 15 May 2024 14:57:06 -0400 Subject: [PATCH 07/17] Rename test_simulator.py --- .../{simulator/test_native_interpreter.py => test_simulator.py} | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename test/unit_tests/autoqasm/{simulator/test_native_interpreter.py => test_simulator.py} (100%) diff --git a/test/unit_tests/autoqasm/simulator/test_native_interpreter.py b/test/unit_tests/autoqasm/test_simulator.py similarity index 100% rename from test/unit_tests/autoqasm/simulator/test_native_interpreter.py rename to test/unit_tests/autoqasm/test_simulator.py From 122ad2857b0e1c4b6f0bf07ecac2932d968cd5ba Mon Sep 17 00:00:00 2001 From: Ryan Shaffer <3620100+rmshaffer@users.noreply.github.com> Date: Wed, 15 May 2024 15:04:39 -0400 Subject: [PATCH 08/17] Delete unused code --- src/autoqasm/simulator/linalg_utils.py | 39 -------------------------- 1 file changed, 39 deletions(-) diff --git a/src/autoqasm/simulator/linalg_utils.py b/src/autoqasm/simulator/linalg_utils.py index 0b3d50c..5a3bc65 100644 --- a/src/autoqasm/simulator/linalg_utils.py +++ b/src/autoqasm/simulator/linalg_utils.py @@ -32,45 +32,6 @@ def measurement_sample(prob: float, target_count: int) -> tuple[int]: return tuple(basis_states[outcome_idx]) -def measurement_collapse_dm( - dm_tensor: np.ndarray, targets: Iterable[int], outcomes: np.ndarray -) -> np.ndarray: - """_summary_ - - Args: - dm_tensor (np.ndarray): _description_ - targets (Iterable[int]): _description_ - outcomes (np.ndarray): _description_ - - Returns: - np.ndarray: _description_ - """ - # TODO: This needs to be modified to not delete qubits - - # move the target qubit to the front of axes - qubit_count = int(np.log2(dm_tensor.shape[0])) - unused_idxs = [idx for idx in range(qubit_count) if idx not in targets] - unused_idxs = [ - p + i * qubit_count for i in range(2) for p in unused_idxs - ] # convert indices to dm form - target_indx = [ - p + i * qubit_count for i in range(2) for p in targets - ] # convert indices to dm form - permutation = target_indx + unused_idxs - inverse_permutation = np.argsort(permutation) - - # collapse the density matrix based on measuremnt outcome - outcomes = tuple(i for _ in range(2) for i in outcomes) - new_dm_tensor = np.zeros_like(dm_tensor) - new_dm_tensor[outcomes] = np.transpose(dm_tensor, permutation)[outcomes] - new_dm_tensor = np.transpose(new_dm_tensor, inverse_permutation) - - # normalize - new_trace = np.trace(np.reshape(new_dm_tensor, (2**qubit_count, 2**qubit_count))) - new_dm_tensor = new_dm_tensor / new_trace - return new_dm_tensor - - def measurement_collapse_sv( state_vector: np.ndarray, targets: Iterable[int], outcome: np.ndarray ) -> np.ndarray: From 3eea679273a54d8a3495d3ef3be9192797c1fe1d Mon Sep 17 00:00:00 2001 From: Ryan Shaffer <3620100+rmshaffer@users.noreply.github.com> Date: Wed, 15 May 2024 16:42:48 -0400 Subject: [PATCH 09/17] Code coverage for simulator.py --- src/autoqasm/simulator/simulator.py | 28 ---------------------- test/unit_tests/autoqasm/test_simulator.py | 25 +++++++++++++++---- 2 files changed, 21 insertions(+), 32 deletions(-) diff --git a/src/autoqasm/simulator/simulator.py b/src/autoqasm/simulator/simulator.py index d823648..60a9e98 100644 --- a/src/autoqasm/simulator/simulator.py +++ b/src/autoqasm/simulator/simulator.py @@ -12,7 +12,6 @@ # language governing permissions and limitations under the License. from braket.default_simulator import StateVectorSimulator -from braket.default_simulator.openqasm.circuit import Circuit from braket.ir.openqasm import Program as OpenQASMProgram from braket.task_result import AdditionalMetadata, TaskMetadata from braket.tasks import GateModelQuantumTaskResult @@ -85,30 +84,3 @@ def run( additional_metadata=AdditionalMetadata.construct(), measurements=context, ) - - def _validate_input_provided(self, circuit: Circuit) -> None: - """ - Validate that requested circuit has all input parameters provided. - - Args: - circuit (Circuit): IR for the simulator. - - Raises: - NameError: If any the specified input parameters are not provided - """ - for instruction in circuit.instructions: - possible_parameters = "_angle", "_angle_1", "_angle_2" - for parameter_name in possible_parameters: - param = getattr(instruction, parameter_name, None) - if param is not None: - try: - float(param) - except TypeError: - missing_input = param.free_symbols.pop() - raise NameError(f"Missing input variable '{missing_input}'.") - for qubit in instruction.targets: - try: - float(qubit) - except TypeError: - missing_input = qubit.free_symbols.pop() - raise NameError(f"Missing input variable '{missing_input}'.") diff --git a/test/unit_tests/autoqasm/test_simulator.py b/test/unit_tests/autoqasm/test_simulator.py index 41413cb..f56b445 100644 --- a/test/unit_tests/autoqasm/test_simulator.py +++ b/test/unit_tests/autoqasm/test_simulator.py @@ -12,13 +12,30 @@ # language governing permissions and limitations under the License. import pytest +from braket.ir.openqasm import Program as OpenQASMProgram +from braket.tasks import GateModelQuantumTaskResult +from autoqasm.simulator import McmSimulator from autoqasm.simulator.native_interpreter import NativeInterpreter +from autoqasm.simulator.program_context import McmProgramContext from autoqasm.simulator.simulation import Simulation INPUTS_QASM = "test/resources/inputs.qasm" +def test_simulator_run(): + program = OpenQASMProgram(source="OPENQASM 3.0;") + simulator = McmSimulator() + result = simulator.run(program) + assert isinstance(result, GateModelQuantumTaskResult) + + +def test_simulator_create_program_context(): + simulator = McmSimulator() + program_context = simulator.create_program_context() + assert isinstance(program_context, McmProgramContext) + + @pytest.mark.parametrize( "reset_instructions", ( @@ -123,14 +140,14 @@ def test_physical_qubit_identifier(): def test_qubit_register_indexing(): qasm = """ OPENQASM 3.0; + input int index; qubit[2] __qubits__; - int a = 0; - x __qubits__[a+1]; + x __qubits__[index]; h __qubits__[0:1]; - h __qubits__[{a, 1}]; + h __qubits__[{0, index}]; bit[2] __bit_0__ = measure __qubits__; """ - result = NativeInterpreter(Simulation(1, 1, 1)).simulate(qasm) + result = NativeInterpreter(Simulation(1, 1, 1)).simulate(qasm, inputs={"index": 1}) assert result["__bit_0__"] == ["01"] From c75a4de4a5056523107c280ff377336b23e5ba77 Mon Sep 17 00:00:00 2001 From: Ryan Shaffer <3620100+rmshaffer@users.noreply.github.com> Date: Wed, 15 May 2024 16:51:22 -0400 Subject: [PATCH 10/17] Remove unreachable code --- src/autoqasm/simulator/conversion.py | 7 +------ 1 file changed, 1 insertion(+), 6 deletions(-) diff --git a/src/autoqasm/simulator/conversion.py b/src/autoqasm/simulator/conversion.py index acf92e3..31981be 100644 --- a/src/autoqasm/simulator/conversion.py +++ b/src/autoqasm/simulator/conversion.py @@ -29,7 +29,7 @@ @singledispatch def convert_to_output(value: LiteralType) -> Any: - raise TypeError(f"converting {value} to output") + raise NotImplementedError(f"converting {value} to output") @convert_to_output.register(IntegerLiteral) @@ -40,11 +40,6 @@ def _(value): return value.value -@convert_to_output.register(BitstringLiteral) -def _(value): - return np.array(np.binary_repr(value.value, value.width)) - - @convert_to_output.register def _(value: ArrayLiteral): if isinstance(value.values[0], BooleanLiteral): From 1327d4c84c77fd03a0888a6388b905bfc0fa2ea3 Mon Sep 17 00:00:00 2001 From: Ryan Shaffer <3620100+rmshaffer@users.noreply.github.com> Date: Thu, 16 May 2024 16:09:05 -0400 Subject: [PATCH 11/17] Finish code coverage --- src/autoqasm/simulator/program_context.py | 7 +++++-- test/unit_tests/autoqasm/test_simulator.py | 24 ++++++++++++++++------ 2 files changed, 23 insertions(+), 8 deletions(-) diff --git a/src/autoqasm/simulator/program_context.py b/src/autoqasm/simulator/program_context.py index 03f8f30..b7216b8 100644 --- a/src/autoqasm/simulator/program_context.py +++ b/src/autoqasm/simulator/program_context.py @@ -34,6 +34,7 @@ ) from braket.default_simulator.openqasm.program_context import ProgramContext, Table from braket.default_simulator.operation import GateOperation +from sympy import Integer from autoqasm.simulator.conversion import convert_to_output @@ -103,7 +104,9 @@ def _(self, identifier: IndexedIdentifier) -> tuple[int]: # noqa: C901 # validate indices manually, since we use addition instead of indexing to # accommodate symbolic indices for q in target: - if isinstance(q, int) and (relative_index := q - self[name][0]) >= len(self[name]): + if isinstance(q, (int, Integer)) and (relative_index := q - self[name][0]) >= len( + self[name] + ): raise IndexError( f"qubit register index `{relative_index}` out of range for qubit register " f"of length {len(self[name])} `{name}`." @@ -158,7 +161,7 @@ def _get_indices_length( elif isinstance(last_index, DiscreteSet): return len(last_index.values) else: - raise NotImplementedError + raise TypeError(f"tuple indices must be integers or slices, not {type(last_index)}") def get_qubit_size(self, identifier: Union[Identifier, IndexedIdentifier]) -> int: """_summary_ diff --git a/test/unit_tests/autoqasm/test_simulator.py b/test/unit_tests/autoqasm/test_simulator.py index f56b445..993f394 100644 --- a/test/unit_tests/autoqasm/test_simulator.py +++ b/test/unit_tests/autoqasm/test_simulator.py @@ -119,9 +119,8 @@ def test_qubit_index_out_of_range_symbolic(): qasm = """ OPENQASM 3.0; qubit[2] __qubits__; - for int[32] i in [0:3] { - x __qubits__[i]; - } + h __qubits__[2+pi-pi]; + bit[2] __bit_0__ = measure __qubits__; """ with pytest.raises(IndexError, match="qubit register index `2` out of range"): NativeInterpreter(Simulation(2, 1, 1)).simulate(qasm) @@ -151,11 +150,24 @@ def test_qubit_register_indexing(): assert result["__bit_0__"] == ["01"] -def test_qubit_register_indexing_multi_dimensions(): - qasm = """ +@pytest.mark.parametrize("invalid_index", ("1.23", "pi", "{0, pi}")) +def test_qubit_register_invalid_index(invalid_index): + qasm = f""" + OPENQASM 3.0; + qubit[2] __qubits__; + x __qubits__[{invalid_index}]; + bit[2] __bit_0__ = measure __qubits__; + """ + with pytest.raises(TypeError, match="tuple indices must be integers or slices"): + NativeInterpreter(Simulation(1, 1, 1)).simulate(qasm) + + +@pytest.mark.parametrize("multi_dimension_index", ("[0, 1]", "[0][1][2]")) +def test_qubit_register_indexing_multi_dimensions(multi_dimension_index): + qasm = f""" OPENQASM 3.0; qubit[2] __qubits__; - x __qubits__[0, 1]; + x __qubits__{multi_dimension_index}; """ with pytest.raises(IndexError, match="Cannot index multiple dimensions for qubits."): NativeInterpreter(Simulation(2, 1, 1)).simulate(qasm) From ee12426a147183fcc483fb9f2780e13c51bba64c Mon Sep 17 00:00:00 2001 From: Ryan Shaffer <3620100+rmshaffer@users.noreply.github.com> Date: Thu, 16 May 2024 16:31:22 -0400 Subject: [PATCH 12/17] Finish docstrings for simulator methods --- src/autoqasm/simulator/linalg_utils.py | 22 +++++++------ src/autoqasm/simulator/native_interpreter.py | 14 ++++---- src/autoqasm/simulator/program_context.py | 34 +++++++++++--------- src/autoqasm/simulator/simulation.py | 16 ++++----- 4 files changed, 47 insertions(+), 39 deletions(-) diff --git a/src/autoqasm/simulator/linalg_utils.py b/src/autoqasm/simulator/linalg_utils.py index 5a3bc65..87e8565 100644 --- a/src/autoqasm/simulator/linalg_utils.py +++ b/src/autoqasm/simulator/linalg_utils.py @@ -17,15 +17,17 @@ import numpy as np -def measurement_sample(prob: float, target_count: int) -> tuple[int]: - """_summary_ +def measurement_sample(prob: Iterable[float], target_count: int) -> tuple[int]: + """Draws a random measurement sample. Args: - prob (float): _description_ - target_count (int): _description_ + prob (Iterable[float]): Probabilities of each measurement outcome. + Possible measurement outcomes range from 0 to 2**target_count-1, + meaning that len(prob) must be equal to 2**target_count. + target_count (int): Number of bits in the resulting measurement. Returns: - tuple[int]: _description_ + tuple[int]: Measurement outcomes 0 or 1 for each of the target_count bits. """ basis_states = np.array(list(itertools.product([0, 1], repeat=target_count))) outcome_idx = np.random.choice(list(range(2**target_count)), p=prob) @@ -35,15 +37,15 @@ def measurement_sample(prob: float, target_count: int) -> tuple[int]: def measurement_collapse_sv( state_vector: np.ndarray, targets: Iterable[int], outcome: np.ndarray ) -> np.ndarray: - """_summary_ + """Collapses the state vector according to the given measurement outcome. Args: - state_vector (np.ndarray): _description_ - targets (Iterable[int]): _description_ - outcome (np.ndarray): _description_ + state_vector (np.ndarray): The state vector prior to measurement. + targets (Iterable[int]): The qubit indices that were measured. + outcome (np.ndarray): Array of measurement outcomes 0 or 1. Returns: - np.ndarray: _description_ + np.ndarray: The collapsed state vector after measurement. """ qubit_count = int(np.log2(state_vector.size)) state_tensor = state_vector.reshape([2] * qubit_count) diff --git a/src/autoqasm/simulator/native_interpreter.py b/src/autoqasm/simulator/native_interpreter.py index 9367f06..7da4307 100644 --- a/src/autoqasm/simulator/native_interpreter.py +++ b/src/autoqasm/simulator/native_interpreter.py @@ -57,16 +57,18 @@ def simulate( is_file: bool = False, shots: int = 1, ) -> dict[str, Any]: - """_summary_ + """Simulates the given program. Args: - source (str): _description_ - inputs (Optional[dict[str, Any]]): _description_. Defaults to None. - is_file (bool): _description_. Defaults to False. - shots (int): _description_. Defaults to 1. + source (str): The OpenQASM source program, or a filename containing the + OpenQASM source program. + inputs (Optional[dict[str, Any]]): The input parameter values to the OpenQASM + source program. Defaults to None. + is_file (bool): Whether `source` is a filename. Defaults to False. + shots (int): Number of shots of the program to simulate. Defaults to 1. Returns: - dict[str, Any]: _description_ + dict[str, Any]: Outputs of the program. """ if inputs: self.context.load_inputs(inputs) diff --git a/src/autoqasm/simulator/program_context.py b/src/autoqasm/simulator/program_context.py index b7216b8..8b1f1da 100644 --- a/src/autoqasm/simulator/program_context.py +++ b/src/autoqasm/simulator/program_context.py @@ -55,10 +55,10 @@ def get_by_identifier(self, identifier: Union[Identifier, IndexedIdentifier]) -> """Convenience method to get an element with a possibly indexed identifier. Args: - identifier (Union[Identifier, IndexedIdentifier]): _description_ + identifier (Union[Identifier, IndexedIdentifier]): The identifier to retrieve. Returns: - tuple[int]: _description_ + tuple[int]: The qubit indices associated with the given identifier. """ if identifier.name.startswith("$"): return (int(identifier.name[1:]),) @@ -70,13 +70,13 @@ def _(self, identifier: IndexedIdentifier) -> tuple[int]: # noqa: C901 corresponding to the elements referenced by the indexed identifier. Args: - identifier (IndexedIdentifier): _description_ + identifier (IndexedIdentifier): The indexed identifier to retrieve. Raises: IndexError: Qubit register index out of range for specified register. Returns: - tuple[int]: _description_ + tuple[int]: The qubit indices associated with the given identifier. """ name = identifier.name.name indices = self.get_qubit_indices(identifier) @@ -117,16 +117,18 @@ def _(self, identifier: IndexedIdentifier) -> tuple[int]: # noqa: C901 def get_qubit_indices( identifier: IndexedIdentifier, ) -> list[IntegerLiteral | RangeDefinition | DiscreteSet]: - """_summary_ + """Gets the qubit indices from a given indexed identifier. Args: - identifier (IndexedIdentifier): _description_ + identifier (IndexedIdentifier): The identifier representing the + qubit indices. Raises: IndexError: Index consists of multiple dimensions. Returns: - list[IntegerLiteral | RangeDefinition | DiscreteSet]: _description_ + list[IntegerLiteral | RangeDefinition | DiscreteSet]: The qubit indices + corresponding to the given indexed identifier. """ primary_index = identifier.indices[0] @@ -164,13 +166,14 @@ def _get_indices_length( raise TypeError(f"tuple indices must be integers or slices, not {type(last_index)}") def get_qubit_size(self, identifier: Union[Identifier, IndexedIdentifier]) -> int: - """_summary_ + """Gets the number of qubit indices for the given identifier. Args: - identifier (Union[Identifier, IndexedIdentifier]): _description_ + identifier (Union[Identifier, IndexedIdentifier]): The identifier representing + the qubit indices. Returns: - int: _description_ + int: The number of qubit indices contained in the given identifier. """ if isinstance(identifier, IndexedIdentifier): indices = self.get_qubit_indices(identifier) @@ -191,25 +194,26 @@ def __init__(self, circuit: Optional[Circuit] = None): self._circuit = circuit or Circuit() def pop_instructions(self) -> list[GateOperation]: - """_summary_ + """Returns the list of instructions and removes them from the context. Returns: - list[GateOperation]: _description_ + list[GateOperation]: The list of instructions from the context. """ instructions = self.circuit.instructions self.circuit.instructions = [] return instructions def add_output(self, output_name: str) -> None: - """_summary_ + """Adds an output with the given name. Args: - output_name (str): _description_ + output_name (str): The output name to add. """ self.outputs[output_name] = [] def save_output_values(self) -> None: - """_summary_""" + """Saves the shot data to the outputs list. If no outputs have been added + explicitly, all symbols in the current scope are added to the outputs list.""" if not self.outputs: self.outputs = { v: [] diff --git a/src/autoqasm/simulator/simulation.py b/src/autoqasm/simulator/simulation.py index 25eba9b..18b0fc4 100644 --- a/src/autoqasm/simulator/simulation.py +++ b/src/autoqasm/simulator/simulation.py @@ -21,10 +21,10 @@ class Simulation(StateVectorSimulation): def add_qubits(self, num_qubits: int) -> None: - """_summary_ + """Adds the given number of qubits to the simulation. Args: - num_qubits (int): _description_ + num_qubits (int): The number of qubits to add. """ expanded_dims = np.expand_dims(self.state_vector, -1) expanded_qubits = np.append( @@ -34,13 +34,13 @@ def add_qubits(self, num_qubits: int) -> None: self._qubit_count += num_qubits def measure(self, targets: tuple[int]) -> tuple[int]: - """_summary_ + """Measures the specified qubits and returns the outcome. Args: - targets (tuple[int]): _description_ + targets (tuple[int]): The qubit indices to measure. Returns: - tuple[int]: _description_ + tuple[int]: The measurement outcomes 0 or 1 for each measured qubit. """ mprob = marginal_probability(self.probabilities, targets) outcome = measurement_sample(mprob, len(targets)) @@ -52,14 +52,14 @@ def measure(self, targets: tuple[int]) -> tuple[int]: return outcome def reset(self) -> None: - """_summary_""" + """Resets the simulation and resets the qubit count to 0.""" self._state_vector = np.array([1], dtype=complex) self._qubit_count = 0 def flip(self, target: int) -> None: - """_summary_ + """Performs a bit flip (PauliX operation) on the specified qubit. Args: - target (int): _description_ + target (int): The qubit index on which to perform a bit flip. """ self.evolve([PauliX([target])]) From e3fc7a035746c0e512c044c56d85ae556209e485 Mon Sep 17 00:00:00 2001 From: Ryan Shaffer <3620100+rmshaffer@users.noreply.github.com> Date: Tue, 21 May 2024 09:56:43 -0400 Subject: [PATCH 13/17] Change build_if_necessary default --- setup.py | 2 +- src/autoqasm/program/program.py | 4 ++-- tox.ini | 2 +- 3 files changed, 4 insertions(+), 4 deletions(-) diff --git a/setup.py b/setup.py index 0cafbff..562c4bc 100644 --- a/setup.py +++ b/setup.py @@ -25,7 +25,7 @@ package_dir={"": "src"}, install_requires=[ # Pin the latest commit of mcm-sim branch of amazon-braket/amazon-braket-sdk-python.git - "amazon-braket-sdk @ git+https://github.com/amazon-braket/amazon-braket-sdk-python.git@da74ee713c200ac0803704bb0b4c87f9d0100682#egg=amazon-braket-sdk", # noqa E501 + "amazon-braket-sdk @ git+https://github.com/amazon-braket/amazon-braket-sdk-python.git@677d8107225098e32a127d3c7a4a8767d63f9d7a#egg=amazon-braket-sdk", # noqa E501 "amazon-braket-default-simulator>=1.23.2", "oqpy~=0.3.5", "diastatic-malt", diff --git a/src/autoqasm/program/program.py b/src/autoqasm/program/program.py index 97809db..d833bfa 100644 --- a/src/autoqasm/program/program.py +++ b/src/autoqasm/program/program.py @@ -121,7 +121,7 @@ def build(self, device: Device | str | None = None) -> Program: def to_ir( self, ir_type: IRType = IRType.OPENQASM, - build_if_necessary: bool = False, + build_if_necessary: bool = True, serialization_properties: SerializationProperties = OpenQASMSerializationProperties(), ) -> str: """Serializes the program into an intermediate representation. @@ -130,7 +130,7 @@ def to_ir( ir_type (IRType): The IRType to use for converting the program to its IR representation. Defaults to IRType.OPENQASM. build_if_necessary (bool): Whether to allow the program to be implicitly - built as a side effect of calling this function. Defaults to False. + built as a side effect of calling this function. Defaults to True. serialization_properties (SerializationProperties): IR serialization configuration. Default to OpenQASMSerializationProperties(). diff --git a/tox.ini b/tox.ini index 572d24e..71656c1 100644 --- a/tox.ini +++ b/tox.ini @@ -133,5 +133,5 @@ commands = [test-deps] deps = # If you need to test on a certain branch, add @ after .git - git+https://github.com/amazon-braket/amazon-braket-sdk-python.git@da74ee713c200ac0803704bb0b4c87f9d0100682 # mcm-sim branch + git+https://github.com/amazon-braket/amazon-braket-sdk-python.git@677d8107225098e32a127d3c7a4a8767d63f9d7a # mcm-sim branch git+https://github.com/amazon-braket/amazon-braket-default-simulator-python.git From 74b817b07043fde56181962fd29d287250854c9c Mon Sep 17 00:00:00 2001 From: Ryan Shaffer <3620100+rmshaffer@users.noreply.github.com> Date: Tue, 21 May 2024 13:18:01 -0400 Subject: [PATCH 14/17] Module-level docstring for simulator --- src/autoqasm/simulator/__init__.py | 19 ++++++++++++++++--- 1 file changed, 16 insertions(+), 3 deletions(-) diff --git a/src/autoqasm/simulator/__init__.py b/src/autoqasm/simulator/__init__.py index b95509e..421a404 100644 --- a/src/autoqasm/simulator/__init__.py +++ b/src/autoqasm/simulator/__init__.py @@ -12,13 +12,26 @@ # language governing permissions and limitations under the License. """ -TODO: Module description +This module contains a local simulator implementation which can be used to +simulate AutoQASM programs at the full OpenQASM 3.0 scope of functionality, +including programs which contain mid-circuit measurement and classical control flow. -TODO: Example usage +The recommended usage of this simulator is by instantiating the Braket LocalSimulator +object with the "autoqasm" backend: .. code-block:: python - # Python code here + import autoqasm as aq + from braket.devices.local_simulator import LocalSimulator + + @aq.main + def bell_state(): + h(0) + cnot(0, 1) + return measure([0, 1]) + + device = LocalSimulator("autoqasm") + result = device.run(bell_state, shots=100).result() """ from .simulator import McmSimulator # noqa: F401 From 14c5e22fb7da6f9160f1a04b5f305130af57a8af Mon Sep 17 00:00:00 2001 From: Ryan Shaffer <3620100+rmshaffer@users.noreply.github.com> Date: Wed, 22 May 2024 11:27:08 -0400 Subject: [PATCH 15/17] Simplify LocalSimulator import path --- README.md | 2 +- examples/1_Getting_started_with_AutoQASM.ipynb | 2 +- examples/2_Expressing_classical_control_flow.ipynb | 2 +- examples/3_1_Iterative_phase_estimation.ipynb | 2 +- examples/3_2_magic_state_distillation.ipynb | 2 +- src/autoqasm/simulator/__init__.py | 2 +- test/unit_tests/autoqasm/test_api.py | 2 +- test/unit_tests/autoqasm/test_parameters.py | 2 +- 8 files changed, 8 insertions(+), 8 deletions(-) diff --git a/README.md b/README.md index 617db7c..02a8350 100644 --- a/README.md +++ b/README.md @@ -100,7 +100,7 @@ AutoQASM includes a simulator which can be accessed using the Amazon Braket loca Let's simulate the `conditional_multi_bell_states` program: ``` -from braket.devices.local_simulator import LocalSimulator +from braket.devices import LocalSimulator device = LocalSimulator("autoqasm") task = device.run(conditional_multi_bell_states, shots=100) diff --git a/examples/1_Getting_started_with_AutoQASM.ipynb b/examples/1_Getting_started_with_AutoQASM.ipynb index 962db45..cd89a7d 100644 --- a/examples/1_Getting_started_with_AutoQASM.ipynb +++ b/examples/1_Getting_started_with_AutoQASM.ipynb @@ -21,7 +21,7 @@ "import matplotlib.pyplot as plt\n", "\n", "# AWS imports: Import Braket SDK modules\n", - "from braket.devices.local_simulator import LocalSimulator\n", + "from braket.devices import LocalSimulator\n", "\n", "# AutoQASM imports\n", "import autoqasm as aq\n", diff --git a/examples/2_Expressing_classical_control_flow.ipynb b/examples/2_Expressing_classical_control_flow.ipynb index 5f2726c..eef0d51 100644 --- a/examples/2_Expressing_classical_control_flow.ipynb +++ b/examples/2_Expressing_classical_control_flow.ipynb @@ -22,7 +22,7 @@ "import matplotlib.pyplot as plt\n", "\n", "# AWS imports: Import Braket SDK modules\n", - "from braket.devices.local_simulator import LocalSimulator\n", + "from braket.devices import LocalSimulator\n", "\n", "# AutoQASM imports\n", "import autoqasm as aq\n", diff --git a/examples/3_1_Iterative_phase_estimation.ipynb b/examples/3_1_Iterative_phase_estimation.ipynb index 72a5b2b..4a8a93c 100644 --- a/examples/3_1_Iterative_phase_estimation.ipynb +++ b/examples/3_1_Iterative_phase_estimation.ipynb @@ -24,7 +24,7 @@ "import matplotlib.pyplot as plt\n", "\n", "# AWS imports: Import Braket SDK modules\n", - "from braket.devices.local_simulator import LocalSimulator\n", + "from braket.devices import LocalSimulator\n", "\n", "# AutoQASM imports\n", "import autoqasm as aq\n", diff --git a/examples/3_2_magic_state_distillation.ipynb b/examples/3_2_magic_state_distillation.ipynb index dfed335..bd5ef5d 100644 --- a/examples/3_2_magic_state_distillation.ipynb +++ b/examples/3_2_magic_state_distillation.ipynb @@ -37,7 +37,7 @@ "from collections import defaultdict\n", "\n", "# AWS imports: Import Braket SDK modules\n", - "from braket.devices.local_simulator import LocalSimulator\n", + "from braket.devices import LocalSimulator\n", "\n", "# AutoQASM imports\n", "import autoqasm as aq\n", diff --git a/src/autoqasm/simulator/__init__.py b/src/autoqasm/simulator/__init__.py index 421a404..f9e317f 100644 --- a/src/autoqasm/simulator/__init__.py +++ b/src/autoqasm/simulator/__init__.py @@ -22,7 +22,7 @@ .. code-block:: python import autoqasm as aq - from braket.devices.local_simulator import LocalSimulator + from braket.devices import LocalSimulator @aq.main def bell_state(): diff --git a/test/unit_tests/autoqasm/test_api.py b/test/unit_tests/autoqasm/test_api.py index 3f4a584..aa932d2 100644 --- a/test/unit_tests/autoqasm/test_api.py +++ b/test/unit_tests/autoqasm/test_api.py @@ -17,7 +17,7 @@ """ import pytest -from braket.devices.local_simulator import LocalSimulator +from braket.devices import LocalSimulator from braket.tasks.local_quantum_task import LocalQuantumTask import autoqasm as aq diff --git a/test/unit_tests/autoqasm/test_parameters.py b/test/unit_tests/autoqasm/test_parameters.py index 9dc36a9..a0e1d24 100644 --- a/test/unit_tests/autoqasm/test_parameters.py +++ b/test/unit_tests/autoqasm/test_parameters.py @@ -16,7 +16,7 @@ import numpy as np import pytest from braket.circuits import FreeParameter -from braket.devices.local_simulator import LocalSimulator +from braket.devices import LocalSimulator from braket.tasks.local_quantum_task import LocalQuantumTask import autoqasm as aq From 52bbd4cd650020a99e657a9a615abac8a3c91561 Mon Sep 17 00:00:00 2001 From: Ryan Shaffer <3620100+rmshaffer@users.noreply.github.com> Date: Wed, 22 May 2024 11:34:38 -0400 Subject: [PATCH 16/17] Add simple test for mid-circuit measurement simulation --- test/unit_tests/autoqasm/test_simulator.py | 20 ++++++++++++++++++++ 1 file changed, 20 insertions(+) diff --git a/test/unit_tests/autoqasm/test_simulator.py b/test/unit_tests/autoqasm/test_simulator.py index 993f394..7322171 100644 --- a/test/unit_tests/autoqasm/test_simulator.py +++ b/test/unit_tests/autoqasm/test_simulator.py @@ -171,3 +171,23 @@ def test_qubit_register_indexing_multi_dimensions(multi_dimension_index): """ with pytest.raises(IndexError, match="Cannot index multiple dimensions for qubits."): NativeInterpreter(Simulation(2, 1, 1)).simulate(qasm) + + +def test_mid_circuit_measurement(): + qasm = """ + OPENQASM 3.0; + qubit[2] __qubits__; + x __qubits__[1]; + bit __bit_0__ = measure __qubits__[1]; + if (__bit_0__) { + x __qubits__[0]; + __bit_0__ = measure __qubits__[0]; + if (__bit_0__) { + x __qubits__[1]; + } + } + bit[2] __bit_1__ = measure __qubits__; + """ + result = NativeInterpreter(Simulation(2, 1, 1)).simulate(qasm) + assert result["__bit_0__"] == [1] + assert result["__bit_1__"] == ["10"] From 5698ee64e08dc90ab1fa1465af4a9d3b4d1da5a5 Mon Sep 17 00:00:00 2001 From: Ryan Shaffer <3620100+rmshaffer@users.noreply.github.com> Date: Wed, 22 May 2024 15:23:17 -0400 Subject: [PATCH 17/17] Update Braket SDK package dependency --- CHANGELOG.md | 2 +- setup.py | 3 +-- tox.ini | 2 +- 3 files changed, 3 insertions(+), 4 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 290f79b..c650698 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,6 +1,6 @@ # Changelog -## v0.1.0 (2024-05-09) +## v0.1.0 (2024-05-22) This is the initial pre-release version of AutoQASM. diff --git a/setup.py b/setup.py index 562c4bc..e375a95 100644 --- a/setup.py +++ b/setup.py @@ -24,8 +24,7 @@ packages=find_namespace_packages(where="src", exclude=("test",)), package_dir={"": "src"}, install_requires=[ - # Pin the latest commit of mcm-sim branch of amazon-braket/amazon-braket-sdk-python.git - "amazon-braket-sdk @ git+https://github.com/amazon-braket/amazon-braket-sdk-python.git@677d8107225098e32a127d3c7a4a8767d63f9d7a#egg=amazon-braket-sdk", # noqa E501 + "amazon-braket-sdk>=1.80.0", "amazon-braket-default-simulator>=1.23.2", "oqpy~=0.3.5", "diastatic-malt", diff --git a/tox.ini b/tox.ini index 71656c1..7aaa3d4 100644 --- a/tox.ini +++ b/tox.ini @@ -133,5 +133,5 @@ commands = [test-deps] deps = # If you need to test on a certain branch, add @ after .git - git+https://github.com/amazon-braket/amazon-braket-sdk-python.git@677d8107225098e32a127d3c7a4a8767d63f9d7a # mcm-sim branch + git+https://github.com/amazon-braket/amazon-braket-sdk-python.git git+https://github.com/amazon-braket/amazon-braket-default-simulator-python.git