From 230de0a437ac6b14a81f30956d3d54e705aa5151 Mon Sep 17 00:00:00 2001 From: Tanuj Khattar Date: Tue, 18 Jun 2024 13:01:36 -0700 Subject: [PATCH 1/6] Support symbolic decomposition for state prep via alias sampling --- .../state_preparation_alias_sampling.py | 40 ++++++++++++------- .../state_preparation_alias_sampling_test.py | 12 +++--- 2 files changed, 32 insertions(+), 20 deletions(-) diff --git a/qualtran/bloqs/state_preparation/state_preparation_alias_sampling.py b/qualtran/bloqs/state_preparation/state_preparation_alias_sampling.py index 45d45457b..7438f4474 100644 --- a/qualtran/bloqs/state_preparation/state_preparation_alias_sampling.py +++ b/qualtran/bloqs/state_preparation/state_preparation_alias_sampling.py @@ -30,7 +30,7 @@ from qualtran import bloq_example, BloqDocSpec, BoundedQUInt, Register, Signature from qualtran._infra.gate_with_registers import total_bits from qualtran.bloqs.arithmetic import LessThanEqual -from qualtran.bloqs.basic_gates import CSwap, Hadamard +from qualtran.bloqs.basic_gates import CSwap, Hadamard, OnEach from qualtran.bloqs.block_encoding.lcu_select_and_prepare import PrepareOracle from qualtran.bloqs.data_loading.qrom import QROM from qualtran.bloqs.state_preparation.prepare_uniform_superposition import ( @@ -209,21 +209,31 @@ def qrom_bloq(self) -> QROM: (self.alternates_bitsize, self.keep_bitsize), ) - def decompose_from_registers( - self, - *, - context: cirq.DecompositionContext, - **quregs: NDArray[cirq.Qid], # type:ignore[type-var] - ) -> Iterator[cirq.OP_TREE]: - yield PrepareUniformSuperposition(self.n_coeff).on(*quregs['selection']) + def build_composite_bloq(self, bb, **soqs): + soqs['selection'] = bb.add( + PrepareUniformSuperposition(self.n_coeff), target=soqs['selection'] + ) if self.mu == 0: - return - selection, less_than_equal = quregs['selection'], quregs['less_than_equal'] - sigma_mu, alt, keep = quregs['sigma_mu'], quregs['alt'], quregs['keep'] - yield cirq.H.on_each(*sigma_mu) - yield self.qrom_bloq.on_registers(selection=selection, target0_=alt, target1_=keep) - yield LessThanEqual(self.mu, self.mu).on(*keep, *sigma_mu, *less_than_equal) - yield CSwap.make_on(ctrl=less_than_equal, x=alt, y=selection) + return soqs + selection, less_than_equal = soqs['selection'], soqs['less_than_equal'] + sigma_mu, alt, keep = soqs['sigma_mu'], soqs['alt'], soqs['keep'] + sigma_mu = bb.add(OnEach(self.mu, Hadamard()), q=sigma_mu) + selection, alt, keep = bb.add( + self.qrom_bloq, selection=selection, target0_=alt, target1_=keep + ) + keep, sigma_mu, less_than_equal = bb.add( + LessThanEqual(self.mu, self.mu), x=keep, y=sigma_mu, target=less_than_equal + ) + less_than_equal, alt, selection = bb.add( + CSwap(self.selection_bitsize), ctrl=less_than_equal, x=alt, y=selection + ) + return { + 'selection': selection, + 'less_than_equal': less_than_equal, + 'sigma_mu': sigma_mu, + 'alt': alt, + 'keep': keep, + } def build_call_graph(self, ssa: 'SympySymbolAllocator') -> Set['BloqCountT']: return { diff --git a/qualtran/bloqs/state_preparation/state_preparation_alias_sampling_test.py b/qualtran/bloqs/state_preparation/state_preparation_alias_sampling_test.py index a777e6161..52472ab5a 100644 --- a/qualtran/bloqs/state_preparation/state_preparation_alias_sampling_test.py +++ b/qualtran/bloqs/state_preparation/state_preparation_alias_sampling_test.py @@ -53,6 +53,8 @@ def test_state_prep_alias_sampling_symb(): # Symbolic T-counts symb_t_counts = int(expected_t_count_expr.subs({L: N, sympy.Symbol(r"\epsilon"): epsilon})) np.testing.assert_allclose(concrete_t_counts, symb_t_counts, rtol=1e-4) + # Ensure the symbolic bloq can decomposes into a composite bloq + _ = bloq.decompose_bloq() def assert_state_preparation_valid_for_coefficient(lcu_coefficients: np.ndarray, epsilon: float): @@ -119,11 +121,11 @@ def test_state_preparation_via_coherent_alias_sampling_diagram(): │ │ │ selection1: ────────target───────In───────────────────×(y)─── │ │ -sigma_mu0: ─────────H────────────┼────────In(y)───────┼────── - │ │ │ -sigma_mu1: ─────────H────────────┼────────In(y)───────┼────── - │ │ │ -sigma_mu2: ─────────H────────────┼────────In(y)───────┼────── +sigma_mu0: ─────────H⨂3──────────┼────────In(y)───────┼────── + │ │ │ │ +sigma_mu1: ─────────H⨂3──────────┼────────In(y)───────┼────── + │ │ │ │ +sigma_mu2: ─────────H⨂3──────────┼────────In(y)───────┼────── │ │ │ alt0: ───────────────────────────QROM_a───┼───────────×(x)─── │ │ │ From 779db2508631550ad225307d7e157737ca1fc359 Mon Sep 17 00:00:00 2001 From: Tanuj Khattar Date: Tue, 18 Jun 2024 13:05:32 -0700 Subject: [PATCH 2/6] Fix pylint --- .../bloqs/state_preparation/state_preparation_alias_sampling.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/qualtran/bloqs/state_preparation/state_preparation_alias_sampling.py b/qualtran/bloqs/state_preparation/state_preparation_alias_sampling.py index 7438f4474..11947afcf 100644 --- a/qualtran/bloqs/state_preparation/state_preparation_alias_sampling.py +++ b/qualtran/bloqs/state_preparation/state_preparation_alias_sampling.py @@ -20,7 +20,7 @@ largest absolute error that one can tolerate in the prepared amplitudes. """ from functools import cached_property -from typing import Iterator, Sequence, Set, Tuple, TYPE_CHECKING, Union +from typing import Sequence, Set, Tuple, TYPE_CHECKING, Union import attrs import cirq From 15ba6b4af6fa85ce17a520a52d0add8c4a0b64f9 Mon Sep 17 00:00:00 2001 From: Tanuj Khattar Date: Tue, 18 Jun 2024 13:13:33 -0700 Subject: [PATCH 3/6] Use assert_valid_bloq_decomposition(bloq) --- .../state_preparation/state_preparation_alias_sampling_test.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/qualtran/bloqs/state_preparation/state_preparation_alias_sampling_test.py b/qualtran/bloqs/state_preparation/state_preparation_alias_sampling_test.py index 52472ab5a..814d344a0 100644 --- a/qualtran/bloqs/state_preparation/state_preparation_alias_sampling_test.py +++ b/qualtran/bloqs/state_preparation/state_preparation_alias_sampling_test.py @@ -54,7 +54,7 @@ def test_state_prep_alias_sampling_symb(): symb_t_counts = int(expected_t_count_expr.subs({L: N, sympy.Symbol(r"\epsilon"): epsilon})) np.testing.assert_allclose(concrete_t_counts, symb_t_counts, rtol=1e-4) # Ensure the symbolic bloq can decomposes into a composite bloq - _ = bloq.decompose_bloq() + assert_valid_bloq_decomposition(bloq) def assert_state_preparation_valid_for_coefficient(lcu_coefficients: np.ndarray, epsilon: float): From f0008b79b7255505ecf10bd13d44b7b9439e8984 Mon Sep 17 00:00:00 2001 From: Tanuj Khattar Date: Wed, 17 Jul 2024 12:34:40 -0700 Subject: [PATCH 4/6] Fix formatting --- .../state_preparation/state_preparation_alias_sampling_test.py | 1 - 1 file changed, 1 deletion(-) diff --git a/qualtran/bloqs/state_preparation/state_preparation_alias_sampling_test.py b/qualtran/bloqs/state_preparation/state_preparation_alias_sampling_test.py index 9d785a442..1189b4390 100644 --- a/qualtran/bloqs/state_preparation/state_preparation_alias_sampling_test.py +++ b/qualtran/bloqs/state_preparation/state_preparation_alias_sampling_test.py @@ -93,7 +93,6 @@ def test_state_prep_alias_sampling_symb(): assert_valid_bloq_decomposition(bloq) - def assert_state_preparation_valid_for_coefficient( lcu_coefficients: np.ndarray, epsilon: float, *, sparse: bool = False, atol: float = 1e-6 ): From b6e844922a1f4a2d9945dc8a3d573b57f4aa2c2c Mon Sep 17 00:00:00 2001 From: Tanuj Khattar Date: Wed, 17 Jul 2024 13:08:40 -0700 Subject: [PATCH 5/6] Update docstrings with args and regenerate notebook --- .../state_preparation_alias_sampling.ipynb | 31 ++++++++-------- .../state_preparation_alias_sampling.py | 36 +++++++++---------- 2 files changed, 31 insertions(+), 36 deletions(-) diff --git a/qualtran/bloqs/state_preparation/state_preparation_alias_sampling.ipynb b/qualtran/bloqs/state_preparation/state_preparation_alias_sampling.ipynb index fb06cda64..3b06aa946 100644 --- a/qualtran/bloqs/state_preparation/state_preparation_alias_sampling.ipynb +++ b/qualtran/bloqs/state_preparation/state_preparation_alias_sampling.ipynb @@ -62,23 +62,6 @@ "selecting `l` uniformly at random and then returning it with probability `keep[l] / 2**mu`;\n", "otherwise returning `alt[l]`.\n", "\n", - "#### Parameters\n", - " - `selection_registers`: The input/output registers to prepare the state on (see Signature).\n", - " - `keep`: The discretized `keep` probabilities for alias sampling.\n", - " - `alt`: The alternate/alias values to swap.\n", - " - `mu`: The number of bits to approximate the `keep` probabilities.\n", - " - `sum_of_unnormalized_probabilities`: The total of the input unnormalized probabilities, i.e., $\\lambda$. This is used as the `PrepareOracle.l1_norm_of_coeffs` property. \n", - "\n", - "Signature:\n", - " selection: The input/output register $|\\ell\\rangle$ of size lg(L) where the desired\n", - " coefficient state is prepared.\n", - " temp: Work space comprised of sub signature:\n", - " - sigma: A mu-sized register containing uniform probabilities for comparison against\n", - " `keep`.\n", - " - alt: A lg(L)-sized register of alternate indices\n", - " - keep: a mu-sized register of probabilities of keeping the initially sampled index.\n", - " - one bit for the result of the comparison.\n", - "\n", "This gate corresponds to the following operations:\n", " - UNIFORM_L on the selection register\n", " - H^mu on the sigma register\n", @@ -90,6 +73,20 @@ "Total space will be (2 * log(L) + 2 mu + 1) work qubits + log(L) ancillas for QROM.\n", "The 1 ancilla in work qubits is for the `LessThanEqualGate` followed by coherent swap.\n", "\n", + "#### Registers\n", + " - `selection`: The input/output register $|\\mathrm{ind}_l\\rangle$ of size lg(L) where the desired coefficient state is prepared. Default name is 'selection' if the builder methods on the class are used. Or else, users can specify custom named registers\n", + " - `sigma_mu`: A mu-sized register containing uniform probabilities for comparison against `keep`.\n", + " - `alt`: A lg(L)-sized register of alternate indices\n", + " - `keep`: a mu-sized register of probabilities of keeping the initially sampled index.\n", + " - `less_than_equal`: one bit for the result of the comparison. \n", + "\n", + "#### Parameters\n", + " - `selection_registers`: The input/output registers to prepare the state on (see Registers section).\n", + " - `keep`: The discretized `keep` probabilities for alias sampling.\n", + " - `alt`: The alternate/alias values to swap.\n", + " - `mu`: The number of bits to approximate the `keep` probabilities.\n", + " - `sum_of_unnormalized_probabilities`: The total of the input unnormalized probabilities, i.e., $\\lambda$. This is used as the `PrepareOracle.l1_norm_of_coeffs` property. \n", + "\n", "#### References\n", " - [Encoding Electronic Spectra in Quantum Circuits with Linear T Complexity](https://arxiv.org/abs/1805.03662). Babbush et. al. (2018). Section III.D. and Figure 11.\n" ] diff --git a/qualtran/bloqs/state_preparation/state_preparation_alias_sampling.py b/qualtran/bloqs/state_preparation/state_preparation_alias_sampling.py index 55f150541..247ebbf7b 100644 --- a/qualtran/bloqs/state_preparation/state_preparation_alias_sampling.py +++ b/qualtran/bloqs/state_preparation/state_preparation_alias_sampling.py @@ -23,7 +23,6 @@ from typing import Sequence, Set, Tuple, TYPE_CHECKING, Union import attrs -import cirq import numpy as np from numpy.typing import NDArray @@ -81,24 +80,6 @@ class StatePreparationAliasSampling(PrepareOracle): selecting `l` uniformly at random and then returning it with probability `keep[l] / 2**mu`; otherwise returning `alt[l]`. - Args: - selection_registers: The input/output registers to prepare the state on (see Signature). - keep: The discretized `keep` probabilities for alias sampling. - alt: The alternate/alias values to swap. - mu: The number of bits to approximate the `keep` probabilities. - sum_of_unnormalized_probabilities: The total of the input unnormalized probabilities, - i.e., $\lambda$. This is used as the `PrepareOracle.l1_norm_of_coeffs` property. - - Signature: - selection: The input/output register $|\ell\rangle$ of size lg(L) where the desired - coefficient state is prepared. - temp: Work space comprised of sub signature: - - sigma: A mu-sized register containing uniform probabilities for comparison against - `keep`. - - alt: A lg(L)-sized register of alternate indices - - keep: a mu-sized register of probabilities of keeping the initially sampled index. - - one bit for the result of the comparison. - This gate corresponds to the following operations: - UNIFORM_L on the selection register - H^mu on the sigma register @@ -110,6 +91,23 @@ class StatePreparationAliasSampling(PrepareOracle): Total space will be (2 * log(L) + 2 mu + 1) work qubits + log(L) ancillas for QROM. The 1 ancilla in work qubits is for the `LessThanEqualGate` followed by coherent swap. + Registers: + selection: The input/output register $|\mathrm{ind}_l\rangle$ of size lg(L) where the desired + coefficient state is prepared. Default name is 'selection' if the builder methods on + the class are used. Or else, users can specify custom named registers + sigma_mu: A mu-sized register containing uniform probabilities for comparison against `keep`. + alt: A lg(L)-sized register of alternate indices + keep: a mu-sized register of probabilities of keeping the initially sampled index. + less_than_equal: one bit for the result of the comparison. + + Args: + selection_registers: The input/output registers to prepare the state on (see Registers section). + keep: The discretized `keep` probabilities for alias sampling. + alt: The alternate/alias values to swap. + mu: The number of bits to approximate the `keep` probabilities. + sum_of_unnormalized_probabilities: The total of the input unnormalized probabilities, + i.e., $\lambda$. This is used as the `PrepareOracle.l1_norm_of_coeffs` property. + References: [Encoding Electronic Spectra in Quantum Circuits with Linear T Complexity](https://arxiv.org/abs/1805.03662). Babbush et. al. (2018). Section III.D. and Figure 11. From d3d7eb207bec72fffd274ce9800d16d0bf24e0b9 Mon Sep 17 00:00:00 2001 From: Tanuj Khattar Date: Mon, 12 Aug 2024 22:43:42 -0400 Subject: [PATCH 6/6] Address nits and add an assertion for 1D state prep --- .../state_preparation_alias_sampling.py | 32 +++++++++++++------ .../state_preparation_alias_sampling_test.py | 1 + qualtran/conftest.py | 1 + 3 files changed, 24 insertions(+), 10 deletions(-) diff --git a/qualtran/bloqs/state_preparation/state_preparation_alias_sampling.py b/qualtran/bloqs/state_preparation/state_preparation_alias_sampling.py index 3962c32a3..06b361a98 100644 --- a/qualtran/bloqs/state_preparation/state_preparation_alias_sampling.py +++ b/qualtran/bloqs/state_preparation/state_preparation_alias_sampling.py @@ -47,7 +47,7 @@ from qualtran.symbolics import bit_length, is_symbolic, Shaped, slen, SymbolicFloat, SymbolicInt if TYPE_CHECKING: - from qualtran import BloqBuilder, SoquetT + from qualtran import BloqBuilder, Soquet, SoquetT from qualtran.resource_counting import BloqCountT, SympySymbolAllocator @@ -121,6 +121,14 @@ class StatePreparationAliasSampling(PrepareOracle): mu: SymbolicInt sum_of_unnormalized_probabilities: SymbolicFloat + def __attrs_post_init__(self): + if not is_symbolic(self.mu) and self.mu <= 0: + raise ValueError(f"{self.mu=} must be greater than 0.") + if len(self.selection_registers) != 1: + raise ValueError( + f"{type(self)} only supports 1D state preparation. Found multiple {self.selection_registers=}." + ) + @classmethod def from_probabilities( cls, unnormalized_probabilities: Sequence[float], *, precision: float = 1.0e-5 @@ -233,14 +241,18 @@ def qrom_bloq(self) -> QROM: (self.alternates_bitsize, self.keep_bitsize), ) - def build_composite_bloq(self, bb, **soqs): - soqs['selection'] = bb.add( - PrepareUniformSuperposition(self.n_coeff), target=soqs['selection'] - ) - if self.mu == 0: - return soqs - selection, less_than_equal = soqs['selection'], soqs['less_than_equal'] - sigma_mu, alt, keep = soqs['sigma_mu'], soqs['alt'], soqs['keep'] + def build_composite_bloq( + self, + bb: 'BloqBuilder', + sigma_mu: 'SoquetT', + alt: 'SoquetT', + keep: 'SoquetT', + less_than_equal: 'Soquet', + **soqs: 'SoquetT', + ): + selection = soqs.pop(self.selection_registers[0].name) + assert not soqs + selection = bb.add(PrepareUniformSuperposition(self.n_coeff), target=selection) sigma_mu = bb.add(OnEach(self.mu, Hadamard()), q=sigma_mu) selection, alt, keep = bb.add( self.qrom_bloq, selection=selection, target0_=alt, target1_=keep @@ -252,7 +264,7 @@ def build_composite_bloq(self, bb, **soqs): CSwap(self.selection_bitsize), ctrl=less_than_equal, x=alt, y=selection ) return { - 'selection': selection, + self.selection_registers[0].name: selection, 'less_than_equal': less_than_equal, 'sigma_mu': sigma_mu, 'alt': alt, diff --git a/qualtran/bloqs/state_preparation/state_preparation_alias_sampling_test.py b/qualtran/bloqs/state_preparation/state_preparation_alias_sampling_test.py index 6e08c2fa9..a38f821b3 100644 --- a/qualtran/bloqs/state_preparation/state_preparation_alias_sampling_test.py +++ b/qualtran/bloqs/state_preparation/state_preparation_alias_sampling_test.py @@ -35,6 +35,7 @@ def test_state_prep_alias_sampling_autotest(bloq_autotester): bloq_autotester(_state_prep_alias) + bloq_autotester(_state_prep_alias_symb) def test_sparse_state_prep_alias_sampling_autotest(bloq_autotester): diff --git a/qualtran/conftest.py b/qualtran/conftest.py index 6545294bf..4ae7b83e4 100644 --- a/qualtran/conftest.py +++ b/qualtran/conftest.py @@ -103,6 +103,7 @@ def assert_bloq_example_serializes_for_pytest(bloq_ex: BloqExample): 'apply_lth_bloq', 'linear_combination_block_encoding', 'phase_block_encoding', + 'state_prep_alias_symb', # cannot serialize Shaped 'sparse_matrix_block_encoding', 'sparse_matrix_symb_block_encoding', 'sparse_state_prep_alias_symb', # cannot serialize Shaped