From 6af901fecf60da45ebbefaf86126010ccff43570 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 26 Oct 2023 15:16:59 +0100 Subject: [PATCH 01/51] add checkpoints to gitignore --- .gitignore | 1 + 1 file changed, 1 insertion(+) diff --git a/.gitignore b/.gitignore index a150e1e5..7b054f48 100644 --- a/.gitignore +++ b/.gitignore @@ -7,3 +7,4 @@ dist manual/build manual/jupyter_execute examples/_build/ +*.ipynb_checkpoints \ No newline at end of file From 3090a0dd8b532f9229a40a09598843ab74152f51 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 26 Oct 2023 15:17:29 +0100 Subject: [PATCH 02/51] add qujax vqe notebook to maintained notebooks --- examples/maintained-notebooks.txt | 1 + 1 file changed, 1 insertion(+) diff --git a/examples/maintained-notebooks.txt b/examples/maintained-notebooks.txt index e1d44cab..f66808d2 100644 --- a/examples/maintained-notebooks.txt +++ b/examples/maintained-notebooks.txt @@ -16,5 +16,6 @@ mapping_example spam_example symbolics_example pytket-qujax_qaoa +pytket-qujax_heisenberg_vqe pytket-qujax-classification ucc_vqe From 68e282b827e908a118676874d006004f1615fb4e Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 26 Oct 2023 15:17:50 +0100 Subject: [PATCH 03/51] fix markdown headings in qujax example --- .../python/pytket-qujax_heisenberg_vqe.py | 14 +- examples/pytket-qujax_heisenberg_vqe.ipynb | 484 +++++++++++++++++- 2 files changed, 491 insertions(+), 7 deletions(-) diff --git a/examples/python/pytket-qujax_heisenberg_vqe.py b/examples/python/pytket-qujax_heisenberg_vqe.py index 3c050c9e..fc840f49 100644 --- a/examples/python/pytket-qujax_heisenberg_vqe.py +++ b/examples/python/pytket-qujax_heisenberg_vqe.py @@ -5,10 +5,12 @@ from pytket.circuit.display import render_circuit_jupyter import matplotlib.pyplot as plt + +# ## Let's start with a TKET circuit + import qujax from pytket.extensions.qujax.qujax_convert import tk_to_qujax -# # Let's start with a tket circuit # We place barriers to stop tket automatically rearranging gates and we also store the number of circuit parameters as we'll need this later. @@ -47,7 +49,7 @@ def get_circuit(n_qubits, depth): circuit, n_params = get_circuit(n_qubits, depth) render_circuit_jupyter(circuit) -# # Now let's invoke qujax +# ## Now let's invoke qujax # The `pytket.extensions.qujax.tk_to_qujax` function will generate a parameters -> statetensor function for us. param_to_st = tk_to_qujax(circuit) @@ -72,7 +74,7 @@ def get_circuit(n_qubits, depth): sample_probs = jnp.square(jnp.abs(statevector)) plt.bar(jnp.arange(statevector.size), sample_probs) -# # Cost function +# ## Cost function # Now we have our `param_to_st` function we are free to define a cost function that acts on bitstrings (e.g. maxcut) or integers by directly wrapping a function around `param_to_st`. However, cost functions defined via quantum Hamiltonians are a bit more involved. # Fortunately, we can encode an Hamiltonian in JAX via the `qujax.get_statetensor_to_expectation_func` function which generates a statetensor -> expected value function for us. @@ -119,7 +121,7 @@ def get_circuit(n_qubits, depth): ) param_to_expectation(new_params) -# # We can now use autodiff for fast, exact gradients within a VQE algorithm +# ## Exact gradients within a VQE algorithm # The `param_to_expectation` function we created is a pure JAX function and outputs a scalar. This means we can pass it to `jax.grad` (or even better `jax.value_and_grad`). cost_and_grad = value_and_grad(param_to_expectation) @@ -128,7 +130,7 @@ def get_circuit(n_qubits, depth): cost_and_grad(params) -# # Now we have all the tools we need to design our VQE! +# ## Now we have all the tools we need to design our VQE! # We'll just use vanilla gradient descent with a constant stepsize @@ -164,7 +166,7 @@ def vqe(init_param, n_steps, stepsize): # Pretty good! -# # `jax.jit` speedup +# ## `jax.jit` speedup # One last thing... We can significantly speed up the VQE above via the `jax.jit`. In our current implementation, the expensive `cost_and_grad` function is compiled to [XLA](https://www.tensorflow.org/xla) and then executed at each call. By invoking `jax.jit` we ensure that the function is compiled only once (on the first call) and then simply executed at each future call - this is much faster! cost_and_grad = jit(cost_and_grad) diff --git a/examples/pytket-qujax_heisenberg_vqe.ipynb b/examples/pytket-qujax_heisenberg_vqe.ipynb index 2ec55022..36f288f4 100644 --- a/examples/pytket-qujax_heisenberg_vqe.ipynb +++ b/examples/pytket-qujax_heisenberg_vqe.ipynb @@ -1 +1,483 @@ -{"cells": [{"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.circuit.display import render_circuit_jupyter\n", "from jax import numpy as jnp, random, vmap, grad, value_and_grad, jit\n", "import matplotlib.pyplot as plt"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import qujax\n", "from pytket.extensions.qujax import tk_to_qujax"]}, {"cell_type": "markdown", "metadata": {}, "source": ["# Let's start with a tket circuit
\n", "We place barriers to stop tket automatically rearranging gates and we also store the number of circuit parameters as we'll need this later."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def get_circuit(n_qubits, depth):\n", " n_params = 2 * n_qubits * (depth + 1)\n", " param = jnp.zeros((n_params,))\n", " circuit = Circuit(n_qubits)\n", " k = 0\n", " for i in range(n_qubits):\n", " circuit.H(i)\n", " for i in range(n_qubits):\n", " circuit.Rx(param[k], i)\n", " k += 1\n", " for i in range(n_qubits):\n", " circuit.Ry(param[k], i)\n", " k += 1\n", " for _ in range(depth):\n", " for i in range(0, n_qubits - 1):\n", " circuit.CZ(i, i + 1)\n", " circuit.add_barrier(range(0, n_qubits))\n", " for i in range(n_qubits):\n", " circuit.Rx(param[k], i)\n", " k += 1\n", " for i in range(n_qubits):\n", " circuit.Ry(param[k], i)\n", " k += 1\n", " return circuit, n_params"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["n_qubits = 4\n", "depth = 2\n", "circuit, n_params = get_circuit(n_qubits, depth)\n", "render_circuit_jupyter(circuit)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["# Now let's invoke qujax
\n", "The `pytket.extensions.qujax.tk_to_qujax` function will generate a parameters -> statetensor function for us."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["param_to_st = tk_to_qujax(circuit)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Let's try it out on some random parameters values. Be aware that's JAX's random number generator requires a `jax.random.PRNGkey` every time it's called - more info on that [here](https://jax.readthedocs.io/en/latest/jax.random.html).
\n", "Be aware that we still have convention where parameters are specified as multiples of $\\pi$ - that is in [0,2]."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["params = random.uniform(random.PRNGKey(0), shape=(n_params,), minval=0., maxval=2.)\n", "statetensor = param_to_st(params)\n", "print(statetensor)\n", "print(statetensor.shape)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Note that this function also has an optional second argument where an initiating `statetensor_in` can be provided. If it is not provided it will default to the all 0s state (as we use here)."]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can obtain statevector by simply calling `.flatten()`"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["statevector = statetensor.flatten()\n", "statevector.shape"]}, {"cell_type": "markdown", "metadata": {}, "source": ["And sampling probabilities by squaring the absolute value of the statevector"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["sample_probs = jnp.square(jnp.abs(statevector))\n", "plt.bar(jnp.arange(statevector.size), sample_probs);"]}, {"cell_type": "markdown", "metadata": {}, "source": ["# Cost function"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now we have our `param_to_st` function we are free to define a cost function that acts on bitstrings (e.g. maxcut) or integers by directly wrapping a function around `param_to_st`. However, cost functions defined via quantum Hamiltonians are a bit more involved.
\n", "Fortunately, we can encode an Hamiltonian in JAX via the `qujax.get_statetensor_to_expectation_func` function which generates a statetensor -> expected value function for us.
\n", "It takes three arguments as input
\n", "- `gate_seq_seq`: A list of string (or array) lists encoding the gates in each term of the Hamiltonian. I.e. `[['X','X'], ['Y','Y'], ['Z','Z']]` corresponds to $H = aX_iX_j + bY_kY_l + cZ_mZ_n$ with qubit indices $i,j,k,l,m,n$ specified in the second argument and coefficients $a,b,c$ specified in the third argument
\n", "- `qubit_inds_seq`: A list of integer lists encoding which qubit indices to apply the aforementioned gates. I.e. `[[0, 1],[0,1],[0,1]]`. Must have the same structure as `gate_seq_seq` above.
\n", "- `coefficients`: A list of floats encoding any coefficients in the Hamiltonian. I.e. `[2.3, 0.8, 1.2]` corresponds to $a=2.3,b=0.8,c=1.2$ above. Must have the same length as the two above arguments."]}, {"cell_type": "markdown", "metadata": {}, "source": ["More specifically let's consider the problem of finding the ground state of the quantum Heisenberg Hamiltonian
\n", "$$ H = \\sum_{i=1}^{n_\\text{qubits}-1} X_i X_{i+1} + Y_i Y_{i+1} + Z_i Z_{i+1}. $$
\n", "As described, we define the Hamiltonian via its gate strings, qubit indices and coefficients."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian_gates = [['X', 'X'], ['Y', 'Y'], ['Z', 'Z']] * (n_qubits - 1)\n", "hamiltonian_qubit_inds = [[int(i), int(i) + 1] for i in jnp.repeat(jnp.arange(n_qubits), 3)]\n", "coefficients = [1.] * len(hamiltonian_qubit_inds)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print('Gates:\\t', hamiltonian_gates)\n", "print('Qubits:\\t', hamiltonian_qubit_inds)\n", "print('Coefficients:\\t', coefficients)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now let's get the Hamiltonian as a pure JAX function"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["st_to_expectation = qujax.get_statetensor_to_expectation_func(hamiltonian_gates,\n", " hamiltonian_qubit_inds,\n", " coefficients)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Let's check it works on the statetensor we've already generated."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["expected_val = st_to_expectation(statetensor)\n", "expected_val"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now let's wrap the `param_to_st` and `st_to_expectation` together to give us an all in one `param_to_expectation` cost function."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["param_to_expectation = lambda param: st_to_expectation(param_to_st(param))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["param_to_expectation(params)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Sanity check that a different, randomly generated set of parameters gives us a new expected value."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["new_params = random.uniform(random.PRNGKey(1), shape=(n_params,), minval=0., maxval=2.)\n", "param_to_expectation(new_params)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["# We can now use autodiff for fast, exact gradients within a VQE algorithm
\n", "The `param_to_expectation` function we created is a pure JAX function and outputs a scalar. This means we can pass it to `jax.grad` (or even better `jax.value_and_grad`)."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["cost_and_grad = value_and_grad(param_to_expectation)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The `cost_and_grad` function returns a tuple with the exact cost value and exact gradient evaluated at the parameters."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["cost_and_grad(params)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["# Now we have all the tools we need to design our VQE!
\n", "We'll just use vanilla gradient descent with a constant stepsize"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def vqe(init_param, n_steps, stepsize):\n", " params = jnp.zeros((n_steps, n_params))\n", " params = params.at[0].set(init_param)\n", " cost_vals = jnp.zeros(n_steps)\n", " cost_vals = cost_vals.at[0].set(param_to_expectation(init_param))\n", " for step in range(1, n_steps):\n", " cost_val, cost_grad = cost_and_grad(params[step - 1])\n", " cost_vals = cost_vals.at[step].set(cost_val)\n", " new_param = params[step - 1] - stepsize * cost_grad\n", " params = params.at[step].set(new_param)\n", " print('Iteration:', step, '\\tCost:', cost_val, end='\\r')\n", " print('\\n')\n", " return params, cost_vals"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Ok enough talking, let's run (and whilst we're at it we'll time it too)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["%time vqe_params, vqe_cost_vals = vqe(params, n_steps=250, stepsize=0.01)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Let's plot the results..."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["plt.plot(vqe_cost_vals)\n", "plt.xlabel('Iteration')\n", "plt.ylabel('Cost');"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Pretty good!"]}, {"cell_type": "markdown", "metadata": {}, "source": ["# `jax.jit` speedup
\n", "One last thing... We can significantly speed up the VQE above via the `jax.jit`. In our current implementation, the expensive `cost_and_grad` function is compiled to [XLA](https://www.tensorflow.org/xla) and then executed at each call. By invoking `jax.jit` we ensure that the function is compiled only once (on the first call) and then simply executed at each future call - this is much faster!"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["cost_and_grad = jit(cost_and_grad)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We'll demonstrate this using the second set of initial parameters we randomly generated (to be sure of no caching)."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["%time new_vqe_params, new_vqe_cost_vals = vqe(new_params, n_steps=250, stepsize=0.01)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["That's some speedup!
\n", "But let's also plot the training to be sure it converged correctly"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["plt.plot(new_vqe_cost_vals)\n", "plt.xlabel('Iteration')\n", "plt.ylabel('Cost');"]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{ + "cells": [ + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "# VQE example with pytket-qujax" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "from jax import numpy as jnp, random, value_and_grad, jit\n", + "from pytket import Circuit\n", + "from pytket.circuit.display import render_circuit_jupyter\n", + "import matplotlib.pyplot as plt" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "import qujax\n", + "from pytket.extensions.qujax.qujax_convert import tk_to_qujax" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "We place barriers to stop tket automatically rearranging gates and we also store the number of circuit parameters as we'll need this later." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "def get_circuit(n_qubits, depth):\n", + " n_params = 2 * n_qubits * (depth + 1)\n", + " param = jnp.zeros((n_params,))\n", + " circuit = Circuit(n_qubits)\n", + " k = 0\n", + " for i in range(n_qubits):\n", + " circuit.H(i)\n", + " for i in range(n_qubits):\n", + " circuit.Rx(param[k], i)\n", + " k += 1\n", + " for i in range(n_qubits):\n", + " circuit.Ry(param[k], i)\n", + " k += 1\n", + " for _ in range(depth):\n", + " for i in range(0, n_qubits - 1):\n", + " circuit.CZ(i, i + 1)\n", + " circuit.add_barrier(range(0, n_qubits))\n", + " for i in range(n_qubits):\n", + " circuit.Rx(param[k], i)\n", + " k += 1\n", + " for i in range(n_qubits):\n", + " circuit.Ry(param[k], i)\n", + " k += 1\n", + " return circuit, n_params" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "n_qubits = 4\n", + "depth = 2\n", + "circuit, n_params = get_circuit(n_qubits, depth)\n", + "render_circuit_jupyter(circuit)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "# Now let's invoke qujax
\n", + "The `pytket.extensions.qujax.tk_to_qujax` function will generate a parameters -> statetensor function for us." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "param_to_st = tk_to_qujax(circuit)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Let's try it out on some random parameters values. Be aware that's JAX's random number generator requires a `jax.random.PRNGkey` every time it's called - more info on that [here](https://jax.readthedocs.io/en/latest/jax.random.html).
\n", + "Be aware that we still have convention where parameters are specified as multiples of $\\pi$ - that is in [0,2]." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "params = random.uniform(random.PRNGKey(0), shape=(n_params,), minval=0.0, maxval=2.0)\n", + "statetensor = param_to_st(params)\n", + "print(statetensor)\n", + "print(statetensor.shape)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Note that this function also has an optional second argument where an initiating `statetensor_in` can be provided. If it is not provided it will default to the all 0s state (as we use here)." + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "We can obtain statevector by simply calling `.flatten()`" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "statevector = statetensor.flatten()\n", + "statevector.shape" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "And sampling probabilities by squaring the absolute value of the statevector" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "sample_probs = jnp.square(jnp.abs(statevector))\n", + "plt.bar(jnp.arange(statevector.size), sample_probs)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "# Cost function" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Now we have our `param_to_st` function we are free to define a cost function that acts on bitstrings (e.g. maxcut) or integers by directly wrapping a function around `param_to_st`. However, cost functions defined via quantum Hamiltonians are a bit more involved.
\n", + "Fortunately, we can encode an Hamiltonian in JAX via the `qujax.get_statetensor_to_expectation_func` function which generates a statetensor -> expected value function for us.
\n", + "It takes three arguments as input
\n", + "- `gate_seq_seq`: A list of string (or array) lists encoding the gates in each term of the Hamiltonian. I.e. `[['X','X'], ['Y','Y'], ['Z','Z']]` corresponds to $H = aX_iX_j + bY_kY_l + cZ_mZ_n$ with qubit indices $i,j,k,l,m,n$ specified in the second argument and coefficients $a,b,c$ specified in the third argument
\n", + "- `qubit_inds_seq`: A list of integer lists encoding which qubit indices to apply the aforementioned gates. I.e. `[[0, 1],[0,1],[0,1]]`. Must have the same structure as `gate_seq_seq` above.
\n", + "- `coefficients`: A list of floats encoding any coefficients in the Hamiltonian. I.e. `[2.3, 0.8, 1.2]` corresponds to $a=2.3,b=0.8,c=1.2$ above. Must have the same length as the two above arguments." + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "More specifically let's consider the problem of finding the ground state of the quantum Heisenberg Hamiltonian
\n", + "$$ H = \\sum_{i=1}^{n_\\text{qubits}-1} X_i X_{i+1} + Y_i Y_{i+1} + Z_i Z_{i+1}. $$
\n", + "As described, we define the Hamiltonian via its gate strings, qubit indices and coefficients." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "hamiltonian_gates = [[\"X\", \"X\"], [\"Y\", \"Y\"], [\"Z\", \"Z\"]] * (n_qubits - 1)\n", + "hamiltonian_qubit_inds = [\n", + " [int(i), int(i) + 1] for i in jnp.repeat(jnp.arange(n_qubits), 3)\n", + "]\n", + "coefficients = [1.0] * len(hamiltonian_qubit_inds)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "print(\"Gates:\\t\", hamiltonian_gates)\n", + "print(\"Qubits:\\t\", hamiltonian_qubit_inds)\n", + "print(\"Coefficients:\\t\", coefficients)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Now let's get the Hamiltonian as a pure JAX function" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "st_to_expectation = qujax.get_statetensor_to_expectation_func(\n", + " hamiltonian_gates, hamiltonian_qubit_inds, coefficients\n", + ")" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Let's check it works on the statetensor we've already generated." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "expected_val = st_to_expectation(statetensor)\n", + "expected_val" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Now let's wrap the `param_to_st` and `st_to_expectation` together to give us an all in one `param_to_expectation` cost function." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "param_to_expectation = lambda param: st_to_expectation(param_to_st(param))" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "param_to_expectation(params)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Sanity check that a different, randomly generated set of parameters gives us a new expected value." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "new_params = random.uniform(\n", + " random.PRNGKey(1), shape=(n_params,), minval=0.0, maxval=2.0\n", + ")\n", + "param_to_expectation(new_params)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "# We can now use autodiff for fast, exact gradients within a VQE algorithm
\n", + "The `param_to_expectation` function we created is a pure JAX function and outputs a scalar. This means we can pass it to `jax.grad` (or even better `jax.value_and_grad`)." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "cost_and_grad = value_and_grad(param_to_expectation)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "The `cost_and_grad` function returns a tuple with the exact cost value and exact gradient evaluated at the parameters." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "cost_and_grad(params)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "# Now we have all the tools we need to design our VQE!
\n", + "We'll just use vanilla gradient descent with a constant stepsize" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "def vqe(init_param, n_steps, stepsize):\n", + " params = jnp.zeros((n_steps, n_params))\n", + " params = params.at[0].set(init_param)\n", + " cost_vals = jnp.zeros(n_steps)\n", + " cost_vals = cost_vals.at[0].set(param_to_expectation(init_param))\n", + " for step in range(1, n_steps):\n", + " cost_val, cost_grad = cost_and_grad(params[step - 1])\n", + " cost_vals = cost_vals.at[step].set(cost_val)\n", + " new_param = params[step - 1] - stepsize * cost_grad\n", + " params = params.at[step].set(new_param)\n", + " print(\"Iteration:\", step, \"\\tCost:\", cost_val, end=\"\\r\")\n", + " print(\"\\n\")\n", + " return params, cost_vals" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Ok enough talking, let's run (and whilst we're at it we'll time it too)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "%time" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "vqe_params, vqe_cost_vals = vqe(params, n_steps=250, stepsize=0.01)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Let's plot the results..." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "plt.plot(vqe_cost_vals)\n", + "plt.xlabel(\"Iteration\")\n", + "plt.ylabel(\"Cost\")" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Pretty good!" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "# `jax.jit` speedup
\n", + "One last thing... We can significantly speed up the VQE above via the `jax.jit`. In our current implementation, the expensive `cost_and_grad` function is compiled to [XLA](https://www.tensorflow.org/xla) and then executed at each call. By invoking `jax.jit` we ensure that the function is compiled only once (on the first call) and then simply executed at each future call - this is much faster!" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "cost_and_grad = jit(cost_and_grad)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "We'll demonstrate this using the second set of initial parameters we randomly generated (to be sure of no caching)." + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "%time" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "new_vqe_params, new_vqe_cost_vals = vqe(new_params, n_steps=250, stepsize=0.01)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "That's some speedup!
\n", + "But let's also plot the training to be sure it converged correctly" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "plt.plot(new_vqe_cost_vals)\n", + "plt.xlabel(\"Iteration\")\n", + "plt.ylabel(\"Cost\")" + ] + } + ], + "metadata": { + "kernelspec": { + "display_name": "Python 3 (ipykernel)", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.10.6" + } + }, + "nbformat": 4, + "nbformat_minor": 4 +} From 4078792c35e03a07b9757dd6478fdd456ab2281d Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 26 Oct 2023 15:25:43 +0100 Subject: [PATCH 04/51] set navigation_with_keys to true --- examples/_config.yml | 1 + 1 file changed, 1 insertion(+) diff --git a/examples/_config.yml b/examples/_config.yml index a392c511..fc6ae17e 100644 --- a/examples/_config.yml +++ b/examples/_config.yml @@ -4,6 +4,7 @@ sphinx: config: html_show_copyright: false html_theme_options: + navigation_with_keys: True logo: image_light: _static/Quantinuum_logo_black.png image_dark: _static/Quantinuum_logo_white.png From 056a126d16cba2cdef91ea3e1bba6234e1cff72d Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 26 Oct 2023 15:38:48 +0100 Subject: [PATCH 05/51] Start resturcturing table of contents --- examples/_toc.yml | 32 +++++++++++++++++++++++--------- 1 file changed, 23 insertions(+), 9 deletions(-) diff --git a/examples/_toc.yml b/examples/_toc.yml index d1f14deb..c1913e8b 100644 --- a/examples/_toc.yml +++ b/examples/_toc.yml @@ -3,28 +3,42 @@ format: jb-book root: README -chapters: +parts: + - caption: TKET backends + chapters: + - file: backends_example + - file: comparing_simulators + - file: Forest_portability_example + - file: creating_backends + - file: qiskit_integration + - caption: pytket-qujax examples + chapters: + - file: pytket-qujax-classification + - file: pytket-qujax_qaoa + - file: pytket-qujax_heisenberg_vqe + - caption: Circuit compilation + chapters: + - file: mapping_example + - file: measurement_reduction_example + - file: contextual_optimization + + + + + - file: ansatz_sequence_example - file: circuit_analysis_example - file: circuit_generation_example - file: compilation_example - file: conditional_gate_example - file: contextual_optimization -- file: creating_backends - file: measurement_reduction_example - file: mapping_example - file: symbolics_example - file: ucc_vqe -- file: pytket-qujax_qaoa - file: benchmarking/README # The following notebooks are not executed -- file: backends_example -- file: comparing_simulators -- file: qiskit_integration -- file: Forest_portability_example -- file: pytket-qujax-classification - file: expectation_value_example - file: entanglement_swapping -- file: pytket-qujax_heisenberg_vqe - file: spam_example - file: tket_benchmarking From 68298c1446035d745d4633a17d7112f3738ffb73 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 26 Oct 2023 15:51:57 +0100 Subject: [PATCH 06/51] Restructure TOC into subsections --- examples/_toc.yml | 47 +++++++++++++++++++++-------------------------- 1 file changed, 21 insertions(+), 26 deletions(-) diff --git a/examples/_toc.yml b/examples/_toc.yml index c1913e8b..be8f2680 100644 --- a/examples/_toc.yml +++ b/examples/_toc.yml @@ -4,6 +4,11 @@ format: jb-book root: README parts: + - caption: Building Quantum Circuits + chapters: + - file: circuit_analysis_example + - file: conditional_gate_example + - file: circuit_generation_example - caption: TKET backends chapters: - file: backends_example @@ -11,34 +16,24 @@ parts: - file: Forest_portability_example - file: creating_backends - file: qiskit_integration - - caption: pytket-qujax examples - chapters: - - file: pytket-qujax-classification - - file: pytket-qujax_qaoa - - file: pytket-qujax_heisenberg_vqe - caption: Circuit compilation chapters: - file: mapping_example - file: measurement_reduction_example - file: contextual_optimization - - - - - -- file: ansatz_sequence_example -- file: circuit_analysis_example -- file: circuit_generation_example -- file: compilation_example -- file: conditional_gate_example -- file: contextual_optimization -- file: measurement_reduction_example -- file: mapping_example -- file: symbolics_example -- file: ucc_vqe -- file: benchmarking/README -# The following notebooks are not executed -- file: expectation_value_example -- file: entanglement_swapping -- file: spam_example -- file: tket_benchmarking + - file: symbolics_example + - file: compilation_example + - file: ansatz_sequence_example + - caption: Algorithm Demos + chapters: + - file: ucc_vqe + - file: pytket-qujax-classification + - file: pytket-qujax_qaoa + - file: pytket-qujax_heisenberg_vqe + - caption: Other + chapters: + - file: tket_benchmarking + - file: benchmarking/README + - file: entanglement_swapping + - file: spam_example + - file: expectation_value_example From 975ee54e3874b7b5e8f46158092c728c513dc86e Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 26 Oct 2023 16:26:11 +0100 Subject: [PATCH 07/51] remove key in config file --- examples/_config.yml | 1 - 1 file changed, 1 deletion(-) diff --git a/examples/_config.yml b/examples/_config.yml index fc6ae17e..a392c511 100644 --- a/examples/_config.yml +++ b/examples/_config.yml @@ -4,7 +4,6 @@ sphinx: config: html_show_copyright: false html_theme_options: - navigation_with_keys: True logo: image_light: _static/Quantinuum_logo_black.png image_dark: _static/Quantinuum_logo_white.png From 7c1b69f986f6d67f28c7e05b26a63abdfd08f537 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 26 Oct 2023 16:27:35 +0100 Subject: [PATCH 08/51] add updated heisenberg notebook --- examples/pytket-qujax_heisenberg_vqe.ipynb | 484 +-------------------- 1 file changed, 1 insertion(+), 483 deletions(-) diff --git a/examples/pytket-qujax_heisenberg_vqe.ipynb b/examples/pytket-qujax_heisenberg_vqe.ipynb index 36f288f4..7cfd274d 100644 --- a/examples/pytket-qujax_heisenberg_vqe.ipynb +++ b/examples/pytket-qujax_heisenberg_vqe.ipynb @@ -1,483 +1 @@ -{ - "cells": [ - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "# VQE example with pytket-qujax" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "from jax import numpy as jnp, random, value_and_grad, jit\n", - "from pytket import Circuit\n", - "from pytket.circuit.display import render_circuit_jupyter\n", - "import matplotlib.pyplot as plt" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "import qujax\n", - "from pytket.extensions.qujax.qujax_convert import tk_to_qujax" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "We place barriers to stop tket automatically rearranging gates and we also store the number of circuit parameters as we'll need this later." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "def get_circuit(n_qubits, depth):\n", - " n_params = 2 * n_qubits * (depth + 1)\n", - " param = jnp.zeros((n_params,))\n", - " circuit = Circuit(n_qubits)\n", - " k = 0\n", - " for i in range(n_qubits):\n", - " circuit.H(i)\n", - " for i in range(n_qubits):\n", - " circuit.Rx(param[k], i)\n", - " k += 1\n", - " for i in range(n_qubits):\n", - " circuit.Ry(param[k], i)\n", - " k += 1\n", - " for _ in range(depth):\n", - " for i in range(0, n_qubits - 1):\n", - " circuit.CZ(i, i + 1)\n", - " circuit.add_barrier(range(0, n_qubits))\n", - " for i in range(n_qubits):\n", - " circuit.Rx(param[k], i)\n", - " k += 1\n", - " for i in range(n_qubits):\n", - " circuit.Ry(param[k], i)\n", - " k += 1\n", - " return circuit, n_params" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "n_qubits = 4\n", - "depth = 2\n", - "circuit, n_params = get_circuit(n_qubits, depth)\n", - "render_circuit_jupyter(circuit)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "# Now let's invoke qujax
\n", - "The `pytket.extensions.qujax.tk_to_qujax` function will generate a parameters -> statetensor function for us." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "param_to_st = tk_to_qujax(circuit)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "Let's try it out on some random parameters values. Be aware that's JAX's random number generator requires a `jax.random.PRNGkey` every time it's called - more info on that [here](https://jax.readthedocs.io/en/latest/jax.random.html).
\n", - "Be aware that we still have convention where parameters are specified as multiples of $\\pi$ - that is in [0,2]." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "params = random.uniform(random.PRNGKey(0), shape=(n_params,), minval=0.0, maxval=2.0)\n", - "statetensor = param_to_st(params)\n", - "print(statetensor)\n", - "print(statetensor.shape)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "Note that this function also has an optional second argument where an initiating `statetensor_in` can be provided. If it is not provided it will default to the all 0s state (as we use here)." - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "We can obtain statevector by simply calling `.flatten()`" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "statevector = statetensor.flatten()\n", - "statevector.shape" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "And sampling probabilities by squaring the absolute value of the statevector" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "sample_probs = jnp.square(jnp.abs(statevector))\n", - "plt.bar(jnp.arange(statevector.size), sample_probs)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "# Cost function" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "Now we have our `param_to_st` function we are free to define a cost function that acts on bitstrings (e.g. maxcut) or integers by directly wrapping a function around `param_to_st`. However, cost functions defined via quantum Hamiltonians are a bit more involved.
\n", - "Fortunately, we can encode an Hamiltonian in JAX via the `qujax.get_statetensor_to_expectation_func` function which generates a statetensor -> expected value function for us.
\n", - "It takes three arguments as input
\n", - "- `gate_seq_seq`: A list of string (or array) lists encoding the gates in each term of the Hamiltonian. I.e. `[['X','X'], ['Y','Y'], ['Z','Z']]` corresponds to $H = aX_iX_j + bY_kY_l + cZ_mZ_n$ with qubit indices $i,j,k,l,m,n$ specified in the second argument and coefficients $a,b,c$ specified in the third argument
\n", - "- `qubit_inds_seq`: A list of integer lists encoding which qubit indices to apply the aforementioned gates. I.e. `[[0, 1],[0,1],[0,1]]`. Must have the same structure as `gate_seq_seq` above.
\n", - "- `coefficients`: A list of floats encoding any coefficients in the Hamiltonian. I.e. `[2.3, 0.8, 1.2]` corresponds to $a=2.3,b=0.8,c=1.2$ above. Must have the same length as the two above arguments." - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "More specifically let's consider the problem of finding the ground state of the quantum Heisenberg Hamiltonian
\n", - "$$ H = \\sum_{i=1}^{n_\\text{qubits}-1} X_i X_{i+1} + Y_i Y_{i+1} + Z_i Z_{i+1}. $$
\n", - "As described, we define the Hamiltonian via its gate strings, qubit indices and coefficients." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "hamiltonian_gates = [[\"X\", \"X\"], [\"Y\", \"Y\"], [\"Z\", \"Z\"]] * (n_qubits - 1)\n", - "hamiltonian_qubit_inds = [\n", - " [int(i), int(i) + 1] for i in jnp.repeat(jnp.arange(n_qubits), 3)\n", - "]\n", - "coefficients = [1.0] * len(hamiltonian_qubit_inds)" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "print(\"Gates:\\t\", hamiltonian_gates)\n", - "print(\"Qubits:\\t\", hamiltonian_qubit_inds)\n", - "print(\"Coefficients:\\t\", coefficients)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "Now let's get the Hamiltonian as a pure JAX function" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "st_to_expectation = qujax.get_statetensor_to_expectation_func(\n", - " hamiltonian_gates, hamiltonian_qubit_inds, coefficients\n", - ")" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "Let's check it works on the statetensor we've already generated." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "expected_val = st_to_expectation(statetensor)\n", - "expected_val" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "Now let's wrap the `param_to_st` and `st_to_expectation` together to give us an all in one `param_to_expectation` cost function." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "param_to_expectation = lambda param: st_to_expectation(param_to_st(param))" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "param_to_expectation(params)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "Sanity check that a different, randomly generated set of parameters gives us a new expected value." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "new_params = random.uniform(\n", - " random.PRNGKey(1), shape=(n_params,), minval=0.0, maxval=2.0\n", - ")\n", - "param_to_expectation(new_params)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "# We can now use autodiff for fast, exact gradients within a VQE algorithm
\n", - "The `param_to_expectation` function we created is a pure JAX function and outputs a scalar. This means we can pass it to `jax.grad` (or even better `jax.value_and_grad`)." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "cost_and_grad = value_and_grad(param_to_expectation)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "The `cost_and_grad` function returns a tuple with the exact cost value and exact gradient evaluated at the parameters." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "cost_and_grad(params)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "# Now we have all the tools we need to design our VQE!
\n", - "We'll just use vanilla gradient descent with a constant stepsize" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "def vqe(init_param, n_steps, stepsize):\n", - " params = jnp.zeros((n_steps, n_params))\n", - " params = params.at[0].set(init_param)\n", - " cost_vals = jnp.zeros(n_steps)\n", - " cost_vals = cost_vals.at[0].set(param_to_expectation(init_param))\n", - " for step in range(1, n_steps):\n", - " cost_val, cost_grad = cost_and_grad(params[step - 1])\n", - " cost_vals = cost_vals.at[step].set(cost_val)\n", - " new_param = params[step - 1] - stepsize * cost_grad\n", - " params = params.at[step].set(new_param)\n", - " print(\"Iteration:\", step, \"\\tCost:\", cost_val, end=\"\\r\")\n", - " print(\"\\n\")\n", - " return params, cost_vals" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "Ok enough talking, let's run (and whilst we're at it we'll time it too)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "%time" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "vqe_params, vqe_cost_vals = vqe(params, n_steps=250, stepsize=0.01)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "Let's plot the results..." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "plt.plot(vqe_cost_vals)\n", - "plt.xlabel(\"Iteration\")\n", - "plt.ylabel(\"Cost\")" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "Pretty good!" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "# `jax.jit` speedup
\n", - "One last thing... We can significantly speed up the VQE above via the `jax.jit`. In our current implementation, the expensive `cost_and_grad` function is compiled to [XLA](https://www.tensorflow.org/xla) and then executed at each call. By invoking `jax.jit` we ensure that the function is compiled only once (on the first call) and then simply executed at each future call - this is much faster!" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "cost_and_grad = jit(cost_and_grad)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "We'll demonstrate this using the second set of initial parameters we randomly generated (to be sure of no caching)." - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "%time" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "new_vqe_params, new_vqe_cost_vals = vqe(new_params, n_steps=250, stepsize=0.01)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "That's some speedup!
\n", - "But let's also plot the training to be sure it converged correctly" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "metadata": {}, - "outputs": [], - "source": [ - "plt.plot(new_vqe_cost_vals)\n", - "plt.xlabel(\"Iteration\")\n", - "plt.ylabel(\"Cost\")" - ] - } - ], - "metadata": { - "kernelspec": { - "display_name": "Python 3 (ipykernel)", - "language": "python", - "name": "python3" - }, - "language_info": { - "codemirror_mode": { - "name": "ipython", - "version": 3 - }, - "file_extension": ".py", - "mimetype": "text/x-python", - "name": "python", - "nbconvert_exporter": "python", - "pygments_lexer": "ipython3", - "version": "3.10.6" - } - }, - "nbformat": 4, - "nbformat_minor": 4 -} +{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# VQE example with pytket-qujax"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from jax import numpy as jnp, random, value_and_grad, jit\n", "from pytket import Circuit\n", "from pytket.circuit.display import render_circuit_jupyter\n", "import matplotlib.pyplot as plt"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Let's start with a TKET circuit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import qujax\n", "from pytket.extensions.qujax.qujax_convert import tk_to_qujax"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We place barriers to stop tket automatically rearranging gates and we also store the number of circuit parameters as we'll need this later."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def get_circuit(n_qubits, depth):\n", " n_params = 2 * n_qubits * (depth + 1)\n", " param = jnp.zeros((n_params,))\n", " circuit = Circuit(n_qubits)\n", " k = 0\n", " for i in range(n_qubits):\n", " circuit.H(i)\n", " for i in range(n_qubits):\n", " circuit.Rx(param[k], i)\n", " k += 1\n", " for i in range(n_qubits):\n", " circuit.Ry(param[k], i)\n", " k += 1\n", " for _ in range(depth):\n", " for i in range(0, n_qubits - 1):\n", " circuit.CZ(i, i + 1)\n", " circuit.add_barrier(range(0, n_qubits))\n", " for i in range(n_qubits):\n", " circuit.Rx(param[k], i)\n", " k += 1\n", " for i in range(n_qubits):\n", " circuit.Ry(param[k], i)\n", " k += 1\n", " return circuit, n_params"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["n_qubits = 4\n", "depth = 2\n", "circuit, n_params = get_circuit(n_qubits, depth)\n", "render_circuit_jupyter(circuit)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Now let's invoke qujax
\n", "The `pytket.extensions.qujax.tk_to_qujax` function will generate a parameters -> statetensor function for us."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["param_to_st = tk_to_qujax(circuit)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Let's try it out on some random parameters values. Be aware that's JAX's random number generator requires a `jax.random.PRNGkey` every time it's called - more info on that [here](https://jax.readthedocs.io/en/latest/jax.random.html).
\n", "Be aware that we still have convention where parameters are specified as multiples of $\\pi$ - that is in [0,2]."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["params = random.uniform(random.PRNGKey(0), shape=(n_params,), minval=0.0, maxval=2.0)\n", "statetensor = param_to_st(params)\n", "print(statetensor)\n", "print(statetensor.shape)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Note that this function also has an optional second argument where an initiating `statetensor_in` can be provided. If it is not provided it will default to the all 0s state (as we use here)."]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can obtain statevector by simply calling `.flatten()`"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["statevector = statetensor.flatten()\n", "statevector.shape"]}, {"cell_type": "markdown", "metadata": {}, "source": ["And sampling probabilities by squaring the absolute value of the statevector"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["sample_probs = jnp.square(jnp.abs(statevector))\n", "plt.bar(jnp.arange(statevector.size), sample_probs)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Cost function"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now we have our `param_to_st` function we are free to define a cost function that acts on bitstrings (e.g. maxcut) or integers by directly wrapping a function around `param_to_st`. However, cost functions defined via quantum Hamiltonians are a bit more involved.
\n", "Fortunately, we can encode an Hamiltonian in JAX via the `qujax.get_statetensor_to_expectation_func` function which generates a statetensor -> expected value function for us.
\n", "It takes three arguments as input
\n", "- `gate_seq_seq`: A list of string (or array) lists encoding the gates in each term of the Hamiltonian. I.e. `[['X','X'], ['Y','Y'], ['Z','Z']]` corresponds to $H = aX_iX_j + bY_kY_l + cZ_mZ_n$ with qubit indices $i,j,k,l,m,n$ specified in the second argument and coefficients $a,b,c$ specified in the third argument
\n", "- `qubit_inds_seq`: A list of integer lists encoding which qubit indices to apply the aforementioned gates. I.e. `[[0, 1],[0,1],[0,1]]`. Must have the same structure as `gate_seq_seq` above.
\n", "- `coefficients`: A list of floats encoding any coefficients in the Hamiltonian. I.e. `[2.3, 0.8, 1.2]` corresponds to $a=2.3,b=0.8,c=1.2$ above. Must have the same length as the two above arguments."]}, {"cell_type": "markdown", "metadata": {}, "source": ["More specifically let's consider the problem of finding the ground state of the quantum Heisenberg Hamiltonian
\n", "$$ H = \\sum_{i=1}^{n_\\text{qubits}-1} X_i X_{i+1} + Y_i Y_{i+1} + Z_i Z_{i+1}. $$
\n", "As described, we define the Hamiltonian via its gate strings, qubit indices and coefficients."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian_gates = [[\"X\", \"X\"], [\"Y\", \"Y\"], [\"Z\", \"Z\"]] * (n_qubits - 1)\n", "hamiltonian_qubit_inds = [\n", " [int(i), int(i) + 1] for i in jnp.repeat(jnp.arange(n_qubits), 3)\n", "]\n", "coefficients = [1.0] * len(hamiltonian_qubit_inds)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"Gates:\\t\", hamiltonian_gates)\n", "print(\"Qubits:\\t\", hamiltonian_qubit_inds)\n", "print(\"Coefficients:\\t\", coefficients)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now let's get the Hamiltonian as a pure JAX function"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["st_to_expectation = qujax.get_statetensor_to_expectation_func(\n", " hamiltonian_gates, hamiltonian_qubit_inds, coefficients\n", ")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Let's check it works on the statetensor we've already generated."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["expected_val = st_to_expectation(statetensor)\n", "expected_val"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now let's wrap the `param_to_st` and `st_to_expectation` together to give us an all in one `param_to_expectation` cost function."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["param_to_expectation = lambda param: st_to_expectation(param_to_st(param))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["param_to_expectation(params)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Sanity check that a different, randomly generated set of parameters gives us a new expected value."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["new_params = random.uniform(\n", " random.PRNGKey(1), shape=(n_params,), minval=0.0, maxval=2.0\n", ")\n", "param_to_expectation(new_params)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Exact gradients within a VQE algorithm
\n", "The `param_to_expectation` function we created is a pure JAX function and outputs a scalar. This means we can pass it to `jax.grad` (or even better `jax.value_and_grad`)."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["cost_and_grad = value_and_grad(param_to_expectation)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The `cost_and_grad` function returns a tuple with the exact cost value and exact gradient evaluated at the parameters."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["cost_and_grad(params)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Now we have all the tools we need to design our VQE!
\n", "We'll just use vanilla gradient descent with a constant stepsize"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def vqe(init_param, n_steps, stepsize):\n", " params = jnp.zeros((n_steps, n_params))\n", " params = params.at[0].set(init_param)\n", " cost_vals = jnp.zeros(n_steps)\n", " cost_vals = cost_vals.at[0].set(param_to_expectation(init_param))\n", " for step in range(1, n_steps):\n", " cost_val, cost_grad = cost_and_grad(params[step - 1])\n", " cost_vals = cost_vals.at[step].set(cost_val)\n", " new_param = params[step - 1] - stepsize * cost_grad\n", " params = params.at[step].set(new_param)\n", " print(\"Iteration:\", step, \"\\tCost:\", cost_val, end=\"\\r\")\n", " print(\"\\n\")\n", " return params, cost_vals"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Ok enough talking, let's run (and whilst we're at it we'll time it too)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["%time"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["vqe_params, vqe_cost_vals = vqe(params, n_steps=250, stepsize=0.01)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Let's plot the results..."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["plt.plot(vqe_cost_vals)\n", "plt.xlabel(\"Iteration\")\n", "plt.ylabel(\"Cost\")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Pretty good!"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `jax.jit` speedup
\n", "One last thing... We can significantly speed up the VQE above via the `jax.jit`. In our current implementation, the expensive `cost_and_grad` function is compiled to [XLA](https://www.tensorflow.org/xla) and then executed at each call. By invoking `jax.jit` we ensure that the function is compiled only once (on the first call) and then simply executed at each future call - this is much faster!"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["cost_and_grad = jit(cost_and_grad)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We'll demonstrate this using the second set of initial parameters we randomly generated (to be sure of no caching)."]}, {"cell_type": "markdown", "metadata": {}, "source": ["%time"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["new_vqe_params, new_vqe_cost_vals = vqe(new_params, n_steps=250, stepsize=0.01)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["That's some speedup!
\n", "But let's also plot the training to be sure it converged correctly"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["plt.plot(new_vqe_cost_vals)\n", "plt.xlabel(\"Iteration\")\n", "plt.ylabel(\"Cost\")"]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file From d16b441d0e17fcee41da8db1e2b971f52a8c30b0 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 26 Oct 2023 17:20:16 +0100 Subject: [PATCH 09/51] clean up headings in comparing_simulators example --- examples/comparing_simulators.ipynb | 2 +- examples/python/comparing_simulators.py | 22 +++++++++++----------- 2 files changed, 12 insertions(+), 12 deletions(-) diff --git a/examples/comparing_simulators.ipynb b/examples/comparing_simulators.ipynb index a571e208..60405fcd 100644 --- a/examples/comparing_simulators.ipynb +++ b/examples/comparing_simulators.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Comparison of the simulators available through tket"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In this tutorial, we will focus on:
\n", "- exploring the wide array of simulators available through the extension modules for `pytket`;
\n", "- comparing their unique features and capabilities."]}, {"cell_type": "markdown", "metadata": {}, "source": ["This example assumes the reader is familiar with the basics of circuit construction and evaluation.
\n", "
\n", "To run every option in this example, you will need `pytket`, `pytket-qiskit`, `pytket-pyquil`, `pytket-qsharp`, `pytket-qulacs`, and `pytket-projectq`.
\n", "
\n", "With the number of simulator `Backend`s available across the `pytket` extension modules, we are often asked why to use one over another. Surely, any two simulators are equivalent if they are able to sample the circuits in the same way, right? Not quite. In this notebook we go through each of the simulators in turn and describe what sets them apart from others and how to make use of any unique features.
\n", "
\n", "But first, to demonstrate the significant overlap in functionality, we'll just give some examples of common usage for different types of backends."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Sampling simulator usage"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.extensions.qiskit import AerBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Define a circuit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3, 3)\n", "c.Ry(0.7, 0)\n", "c.CX(0, 1)\n", "c.X(2)\n", "c.measure_all()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run on the backend:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()\n", "c = backend.get_compiled_circuit(c)\n", "handle = backend.process_circuit(c, n_shots=2000)\n", "counts = backend.get_result(handle).get_counts()\n", "print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Statevector simulator usage"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.extensions.qiskit import AerStateBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Build a quantum state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3)\n", "c.H(0).CX(0, 1)\n", "c.Rz(0.3, 0)\n", "c.Rz(-0.3, 1)\n", "c.Ry(0.8, 2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Examine the statevector:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerStateBackend()\n", "c = backend.get_compiled_circuit(c)\n", "handle = backend.process_circuit(c)\n", "state = backend.get_result(handle).get_state()\n", "print(state)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Expectation value usage"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit, Qubit\n", "from pytket.extensions.qiskit import AerBackend, AerStateBackend\n", "from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Build a quantum state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3)\n", "c.H(0).CX(0, 1)\n", "c.Rz(0.3, 0)\n", "c.Rz(-0.3, 1)\n", "c.Ry(0.8, 2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Define the measurement operator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["xxi = QubitPauliString({Qubit(0): Pauli.X, Qubit(1): Pauli.X})\n", "zzz = QubitPauliString({Qubit(0): Pauli.Z, Qubit(1): Pauli.Z, Qubit(2): Pauli.Z})\n", "op = QubitPauliOperator({xxi: -1.8, zzz: 0.7})"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run on the backend:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()\n", "c = backend.get_compiled_circuit(c)\n", "exp = backend.get_operator_expectation_value(c, op)\n", "print(exp)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `pytket.extensions.qiskit.AerBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["`AerBackend` wraps up the `qasm_simulator` from the Qiskit Aer package. It supports an extremely flexible set of circuits and uses many effective simulation methods making it a great all-purpose sampling simulator.
\n", "
\n", "Unique features:
\n", "- supports mid-circuit measurement and OpenQASM-style conditional gates;
\n", "- encompasses a variety of underlying simulation methods and automatically selects the best one for each circuit (including statevector, density matrix, (extended) stabilizer and matrix product state);
\n", "- can be provided with a `qiskit.providers.Aer.noise.NoiseModel` on instantiation to perform a noisy simulation."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Useful features:
\n", "- support for fast expectation value calculations according to `QubitPauliString`s or `QubitPauliOperator`s."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.extensions.qiskit import AerBackend\n", "from itertools import combinations\n", "from qiskit.providers.aer.noise import NoiseModel, depolarizing_error"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Quantum teleportation circuit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit()\n", "alice = c.add_q_register(\"a\", 2)\n", "bob = c.add_q_register(\"b\", 1)\n", "data = c.add_c_register(\"d\", 2)\n", "final = c.add_c_register(\"f\", 1)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Start in an interesting state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.Rx(0.3, alice[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Set up a Bell state between Alice and Bob:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.H(alice[1]).CX(alice[1], bob[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Measure Alice's qubits in the Bell basis:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.CX(alice[0], alice[1]).H(alice[0])\n", "c.Measure(alice[0], data[0])\n", "c.Measure(alice[1], data[1])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Correct Bob's qubit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.X(bob[0], condition_bits=[data[0], data[1]], condition_value=1)\n", "c.X(bob[0], condition_bits=[data[0], data[1]], condition_value=3)\n", "c.Z(bob[0], condition_bits=[data[0], data[1]], condition_value=2)\n", "c.Z(bob[0], condition_bits=[data[0], data[1]], condition_value=3)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Measure Bob's qubit to observe the interesting state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.Measure(bob[0], final[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Set up a noisy simulator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["model = NoiseModel()\n", "dep_err = depolarizing_error(0.04, 2)\n", "for i, j in combinations(range(3), r=2):\n", " model.add_quantum_error(dep_err, [\"cx\"], [i, j])\n", " model.add_quantum_error(dep_err, [\"cx\"], [j, i])\n", "backend = AerBackend(noise_model=model)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run circuit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = backend.get_compiled_circuit(c)\n", "handle = backend.process_circuit(c, n_shots=2000)\n", "result = backend.get_result(handle)\n", "counts = result.get_counts([final[0]])\n", "print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `pytket.extensions.qiskit.AerStateBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["`AerStateBackend` provides access to Qiskit Aer's `statevector_simulator`. It supports a similarly large gate set and has competitive speed for statevector simulations.
\n", "
\n", "Useful features:
\n", "- no dependency on external executables, making it easy to install and run on any computer;
\n", "- support for fast expectation value calculations according to `QubitPauliString`s or `QubitPauliOperator`s."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `pytket.extensions.qiskit.AerUnitaryBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Finishing the set of simulators from Qiskit Aer, `AerUnitaryBackend` captures the `unitary_simulator`, allowing for the entire unitary of a pure quantum process to be calculated. This is especially useful for testing small subcircuits that will be used many times in a larger computation.
\n", "
\n", "Unique features:
\n", "- provides the full unitary matrix for a pure quantum circuit."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.extensions.qiskit import AerUnitaryBackend\n", "from pytket.predicates import NoClassicalControlPredicate"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Define a simple quantum incrementer:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3)\n", "c.CCX(2, 1, 0)\n", "c.CX(2, 1)\n", "c.X(2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Examine the unitary:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerUnitaryBackend()\n", "c = backend.get_compiled_circuit(c)\n", "result = backend.run_circuit(c)\n", "unitary = result.get_unitary()\n", "print(unitary.round(1).real)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `pytket.extensions.pyquil.ForestBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Whilst it can, with suitable credentials, be used to access the Rigetti QPUs, the `ForestBackend` also features a simulator mode which turns it into a noiseless sampling simulator that matches the constraints of the simulated device (e.g. the same gate set, restricted connectivity, measurement model, etc.). This is useful when playing around with custom compilation strategies to ensure that your final circuits are suitable to run on the device and for checking that your overall program works fine before you invest in reserving a QPU.
\n", "
\n", "Unique features:
\n", "- faithful recreation of the circuit constraints of Rigetti QPUs."]}, {"cell_type": "markdown", "metadata": {}, "source": ["If trying to use the `ForestBackend` locally (i.e. not on a Rigetti QMI), you will need to have `quilc` and `qvm` running as separate processes in server mode. One easy way of doing this is with `docker` (see the `quilc` and `qvm` documentation for alternative methods of running them):
\n", "`docker run --rm -it -p 5555:5555 rigetti/quilc -R`
\n", "`docker run --rm -it -p 5000:5000 rigetti/qvm -S`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `pytket.extensions.pyquil.ForestStateBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The Rigetti `pyquil` package also provides the `WavefunctionSimulator`, which we present as the `ForestStateBackend`. Functionally, it is very similar to the `AerStateBackend` so can be used interchangeably. It does require that `quilc` and `qvm` are running as separate processes when not running on a Rigetti QMI.
\n", "
\n", "Useful features:
\n", "- support for fast expectation value calculations according to `QubitPauliString`s or Hermitian `QubitPauliOperator`s."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `pytket.extensions.qsharp.QsharpSimulatorBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The `QsharpSimulatorBackend` is another basic sampling simulator that is interchangeable with others, using the Microsoft QDK simulator. Note that the `pytket-qsharp` package is dependent on the `dotnet` SDK and `iqsharp` tool. Please consult the `pytket-qsharp` installation instructions for recommendations."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `pytket.extensions.qsharp.QsharpToffoliSimulatorBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Toffoli circuits form a strict fragment of quantum circuits and can be efficiently simulated. The `QsharpToffoliSimulatorBackend` can only operate on these circuits, but scales much better with system size than regular simulators.
\n", "
\n", "Unique features:
\n", "- efficient simulation of Toffoli circuits."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.extensions.qsharp import QsharpToffoliSimulatorBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Define a circuit - start in a basis state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3)\n", "c.X(0).X(2)\n", "# Define a circuit - incrementer\n", "c.CCX(2, 1, 0)\n", "c.CX(2, 1)\n", "c.X(2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run on the backend:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = QsharpToffoliSimulatorBackend()\n", "c = backend.get_compiled_circuit(c)\n", "handle = backend.process_circuit(c, n_shots=10)\n", "counts = backend.get_result(handle).get_counts()\n", "print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `pytket.extensions.qsharp.QsharpEstimatorBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The `QsharpEstimatorBackend` is not strictly a simulator, as it doesn't model the state of the quantum system and try to identify the final state, but instead analyses the circuit to estimate the required resources to run it. It does not support any of the regular outcome types (e.g. shots, counts, statevector), just the summary of the estimated resources.
\n", "
\n", "Unique features:
\n", "- estimates resources to perform the circuit, without actually simulating/running it."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit"]}, {"cell_type": "markdown", "metadata": {}, "source": ["from pytket.extensions.qsharp import QsharpEstimatorBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Define a circuit - start in a basis state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3)\n", "c.X(0).X(2)\n", "# Define a circuit - incrementer\n", "c.CCX(2, 1, 0)\n", "c.CX(2, 1)\n", "c.X(2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run on the backend:"]}, {"cell_type": "markdown", "metadata": {}, "source": ["(disabled because of https://github.com/CQCL/pytket-qsharp/issues/37)
\n", "backend = QsharpEstimatorBackend()
\n", "c = backend.get_compiled_circuit(c)
\n", "handle = backend.process_circuit(c, n_shots=10)
\n", "resources = backend.get_resources(handle)
\n", "print(resources)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `pytket.extensions.qulacs.QulacsBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The `QulacsBackend` is an all-purpose simulator with both sampling and statevector modes, using the basic CPU simulator from Qulacs.
\n", "
\n", "Unique features:
\n", "- supports both sampling (shots/counts) and complete statevector outputs."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Useful features:
\n", "- support for fast expectation value calculations according to `QubitPauliString`s or Hermitian `QubitPauliOperator`s."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `pytket.extensions.qulacs.QulacsGPUBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["If the GPU version of Qulacs is installed, the `QulacsGPUBackend` will use that to benefit from even faster speeds. It is very easy to get started with using a GPU, as it only requires a CUDA installation and the `qulacs-gpu` package from `pip`. Functionally, it is identical to the `QulacsBackend`, but potentially faster if you have GPU resources available.
\n", "
\n", "Unique features:
\n", "- GPU support for very fast simulation."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `pytket.extensions.projectq.ProjectQBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["ProjectQ is a popular quantum circuit simulator, thanks to its availability and ease of use. It provides a similar level of performance and features to `AerStateBackend`.
\n", "
\n", "Useful features:
\n", "- support for fast expectation value calculations according to `QubitPauliString`s or Hermitian `QubitPauliOperator`s."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Comparison of the simulators available through tket"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In this tutorial, we will focus on:
\n", "- exploring the wide array of simulators available through the extension modules for `pytket`;
\n", "- comparing their unique features and capabilities."]}, {"cell_type": "markdown", "metadata": {}, "source": ["This example assumes the reader is familiar with the basics of circuit construction and evaluation.
\n", "
\n", "To run every option in this example, you will need `pytket`, `pytket-qiskit`, `pytket-pyquil`, `pytket-qsharp`, `pytket-qulacs`, and `pytket-projectq`.
\n", "
\n", "With the number of simulator `Backend`s available across the `pytket` extension modules, we are often asked why to use one over another. Surely, any two simulators are equivalent if they are able to sample the circuits in the same way, right? Not quite. In this notebook we go through each of the simulators in turn and describe what sets them apart from others and how to make use of any unique features.
\n", "
\n", "But first, to demonstrate the significant overlap in functionality, we'll just give some examples of common usage for different types of backends."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Sampling simulator usage"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.extensions.qiskit import AerBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Define a circuit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3, 3)\n", "c.Ry(0.7, 0)\n", "c.CX(0, 1)\n", "c.X(2)\n", "c.measure_all()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run on the backend:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()\n", "c = backend.get_compiled_circuit(c)\n", "handle = backend.process_circuit(c, n_shots=2000)\n", "counts = backend.get_result(handle).get_counts()\n", "print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Statevector simulator usage"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.extensions.qiskit import AerStateBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Build a quantum state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3)\n", "c.H(0).CX(0, 1)\n", "c.Rz(0.3, 0)\n", "c.Rz(-0.3, 1)\n", "c.Ry(0.8, 2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Examine the statevector:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerStateBackend()\n", "c = backend.get_compiled_circuit(c)\n", "handle = backend.process_circuit(c)\n", "state = backend.get_result(handle).get_state()\n", "print(state)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Expectation value usage"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit, Qubit\n", "from pytket.extensions.qiskit import AerBackend, AerStateBackend\n", "from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Build a quantum state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3)\n", "c.H(0).CX(0, 1)\n", "c.Rz(0.3, 0)\n", "c.Rz(-0.3, 1)\n", "c.Ry(0.8, 2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Define the measurement operator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["xxi = QubitPauliString({Qubit(0): Pauli.X, Qubit(1): Pauli.X})\n", "zzz = QubitPauliString({Qubit(0): Pauli.Z, Qubit(1): Pauli.Z, Qubit(2): Pauli.Z})\n", "op = QubitPauliOperator({xxi: -1.8, zzz: 0.7})"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run on the backend:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()\n", "c = backend.get_compiled_circuit(c)\n", "exp = backend.get_operator_expectation_value(c, op)\n", "print(exp)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `AerBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["`AerBackend` wraps up the `qasm_simulator` from the Qiskit Aer package. It supports an extremely flexible set of circuits and uses many effective simulation methods making it a great all-purpose sampling simulator.
\n", "
\n", "Unique features:
\n", "- supports mid-circuit measurement and OpenQASM-style conditional gates;
\n", "- encompasses a variety of underlying simulation methods and automatically selects the best one for each circuit (including statevector, density matrix, (extended) stabilizer and matrix product state);
\n", "- can be provided with a `qiskit.providers.Aer.noise.NoiseModel` on instantiation to perform a noisy simulation."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Useful features:
\n", "- support for fast expectation value calculations according to `QubitPauliString`s or `QubitPauliOperator`s."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.extensions.qiskit import AerBackend\n", "from itertools import combinations\n", "from qiskit.providers.aer.noise import NoiseModel, depolarizing_error"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Quantum teleportation circuit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit()\n", "alice = c.add_q_register(\"a\", 2)\n", "bob = c.add_q_register(\"b\", 1)\n", "data = c.add_c_register(\"d\", 2)\n", "final = c.add_c_register(\"f\", 1)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Start in an interesting state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.Rx(0.3, alice[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Set up a Bell state between Alice and Bob:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.H(alice[1]).CX(alice[1], bob[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Measure Alice's qubits in the Bell basis:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.CX(alice[0], alice[1]).H(alice[0])\n", "c.Measure(alice[0], data[0])\n", "c.Measure(alice[1], data[1])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Correct Bob's qubit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.X(bob[0], condition_bits=[data[0], data[1]], condition_value=1)\n", "c.X(bob[0], condition_bits=[data[0], data[1]], condition_value=3)\n", "c.Z(bob[0], condition_bits=[data[0], data[1]], condition_value=2)\n", "c.Z(bob[0], condition_bits=[data[0], data[1]], condition_value=3)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Measure Bob's qubit to observe the interesting state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.Measure(bob[0], final[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Set up a noisy simulator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["model = NoiseModel()\n", "dep_err = depolarizing_error(0.04, 2)\n", "for i, j in combinations(range(3), r=2):\n", " model.add_quantum_error(dep_err, [\"cx\"], [i, j])\n", " model.add_quantum_error(dep_err, [\"cx\"], [j, i])\n", "backend = AerBackend(noise_model=model)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run circuit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = backend.get_compiled_circuit(c)\n", "handle = backend.process_circuit(c, n_shots=2000)\n", "result = backend.get_result(handle)\n", "counts = result.get_counts([final[0]])\n", "print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `AerStateBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["`AerStateBackend` provides access to Qiskit Aer's `statevector_simulator`. It supports a similarly large gate set and has competitive speed for statevector simulations.
\n", "
\n", "Useful features:
\n", "- no dependency on external executables, making it easy to install and run on any computer;
\n", "- support for fast expectation value calculations according to `QubitPauliString`s or `QubitPauliOperator`s."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `AerUnitaryBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Finishing the set of simulators from Qiskit Aer, `AerUnitaryBackend` captures the `unitary_simulator`, allowing for the entire unitary of a pure quantum process to be calculated. This is especially useful for testing small subcircuits that will be used many times in a larger computation.
\n", "
\n", "Unique features:
\n", "- provides the full unitary matrix for a pure quantum circuit."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.extensions.qiskit import AerUnitaryBackend\n", "from pytket.predicates import NoClassicalControlPredicate"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Define a simple quantum incrementer:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3)\n", "c.CCX(2, 1, 0)\n", "c.CX(2, 1)\n", "c.X(2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Examine the unitary:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerUnitaryBackend()\n", "c = backend.get_compiled_circuit(c)\n", "result = backend.run_circuit(c)\n", "unitary = result.get_unitary()\n", "print(unitary.round(1).real)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `ForestBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Whilst it can, with suitable credentials, be used to access the Rigetti QPUs, the `ForestBackend` also features a simulator mode which turns it into a noiseless sampling simulator that matches the constraints of the simulated device (e.g. the same gate set, restricted connectivity, measurement model, etc.). This is useful when playing around with custom compilation strategies to ensure that your final circuits are suitable to run on the device and for checking that your overall program works fine before you invest in reserving a QPU.
\n", "
\n", "Unique features:
\n", "- faithful recreation of the circuit constraints of Rigetti QPUs."]}, {"cell_type": "markdown", "metadata": {}, "source": ["If trying to use the `ForestBackend` locally (i.e. not on a Rigetti QMI), you will need to have `quilc` and `qvm` running as separate processes in server mode. One easy way of doing this is with `docker` (see the `quilc` and `qvm` documentation for alternative methods of running them):
\n", "`docker run --rm -it -p 5555:5555 rigetti/quilc -R`
\n", "`docker run --rm -it -p 5000:5000 rigetti/qvm -S`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `ForestStateBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The Rigetti `pyquil` package also provides the `WavefunctionSimulator`, which we present as the `ForestStateBackend`. Functionally, it is very similar to the `AerStateBackend` so can be used interchangeably. It does require that `quilc` and `qvm` are running as separate processes when not running on a Rigetti QMI.
\n", "
\n", "Useful features:
\n", "- support for fast expectation value calculations according to `QubitPauliString`s or Hermitian `QubitPauliOperator`s."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `QsharpSimulatorBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The `QsharpSimulatorBackend` is another basic sampling simulator that is interchangeable with others, using the Microsoft QDK simulator. Note that the `pytket-qsharp` package is dependent on the `dotnet` SDK and `iqsharp` tool. Please consult the `pytket-qsharp` installation instructions for recommendations."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `QsharpToffoliSimulatorBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Toffoli circuits form a strict fragment of quantum circuits and can be efficiently simulated. The `QsharpToffoliSimulatorBackend` can only operate on these circuits, but scales much better with system size than regular simulators.
\n", "
\n", "Unique features:
\n", "- efficient simulation of Toffoli circuits."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.extensions.qsharp import QsharpToffoliSimulatorBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Define a circuit - start in a basis state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3)\n", "c.X(0).X(2)\n", "# Define a circuit - incrementer\n", "c.CCX(2, 1, 0)\n", "c.CX(2, 1)\n", "c.X(2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run on the backend:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = QsharpToffoliSimulatorBackend()\n", "c = backend.get_compiled_circuit(c)\n", "handle = backend.process_circuit(c, n_shots=10)\n", "counts = backend.get_result(handle).get_counts()\n", "print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `QsharpEstimatorBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The `QsharpEstimatorBackend` is not strictly a simulator, as it doesn't model the state of the quantum system and try to identify the final state, but instead analyses the circuit to estimate the required resources to run it. It does not support any of the regular outcome types (e.g. shots, counts, statevector), just the summary of the estimated resources.
\n", "
\n", "Unique features:
\n", "- estimates resources to perform the circuit, without actually simulating/running it."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit"]}, {"cell_type": "markdown", "metadata": {}, "source": ["from pytket.extensions.qsharp import QsharpEstimatorBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Define a circuit - start in a basis state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3)\n", "c.X(0).X(2)\n", "# Define a circuit - incrementer\n", "c.CCX(2, 1, 0)\n", "c.CX(2, 1)\n", "c.X(2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run on the backend:"]}, {"cell_type": "markdown", "metadata": {}, "source": ["(disabled because of https://github.com/CQCL/pytket-qsharp/issues/37)
\n", "backend = QsharpEstimatorBackend()
\n", "c = backend.get_compiled_circuit(c)
\n", "handle = backend.process_circuit(c, n_shots=10)
\n", "resources = backend.get_resources(handle)
\n", "print(resources)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `QulacsBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The `QulacsBackend` is an all-purpose simulator with both sampling and statevector modes, using the basic CPU simulator from Qulacs.
\n", "
\n", "Unique features:
\n", "- supports both sampling (shots/counts) and complete statevector outputs."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Useful features:
\n", "- support for fast expectation value calculations according to `QubitPauliString`s or Hermitian `QubitPauliOperator`s."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `QulacsGPUBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["If the GPU version of Qulacs is installed, the `QulacsGPUBackend` will use that to benefit from even faster speeds. It is very easy to get started with using a GPU, as it only requires a CUDA installation and the `qulacs-gpu` package from `pip`. Functionally, it is identical to the `QulacsBackend`, but potentially faster if you have GPU resources available.
\n", "
\n", "Unique features:
\n", "- GPU support for very fast simulation."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `ProjectQBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["ProjectQ is a popular quantum circuit simulator, thanks to its availability and ease of use. It provides a similar level of performance and features to `AerStateBackend`.
\n", "
\n", "Useful features:
\n", "- support for fast expectation value calculations according to `QubitPauliString`s or Hermitian `QubitPauliOperator`s."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file diff --git a/examples/python/comparing_simulators.py b/examples/python/comparing_simulators.py index f301dbb5..b07b9352 100644 --- a/examples/python/comparing_simulators.py +++ b/examples/python/comparing_simulators.py @@ -82,7 +82,7 @@ exp = backend.get_operator_expectation_value(c, op) print(exp) -# ## `pytket.extensions.qiskit.AerBackend` +# ## `AerBackend` # `AerBackend` wraps up the `qasm_simulator` from the Qiskit Aer package. It supports an extremely flexible set of circuits and uses many effective simulation methods making it a great all-purpose sampling simulator. # @@ -149,7 +149,7 @@ counts = result.get_counts([final[0]]) print(counts) -# ## `pytket.extensions.qiskit.AerStateBackend` +# ## `AerStateBackend` # `AerStateBackend` provides access to Qiskit Aer's `statevector_simulator`. It supports a similarly large gate set and has competitive speed for statevector simulations. # @@ -157,7 +157,7 @@ # - no dependency on external executables, making it easy to install and run on any computer; # - support for fast expectation value calculations according to `QubitPauliString`s or `QubitPauliOperator`s. -# ## `pytket.extensions.qiskit.AerUnitaryBackend` +# ## `AerUnitaryBackend` # Finishing the set of simulators from Qiskit Aer, `AerUnitaryBackend` captures the `unitary_simulator`, allowing for the entire unitary of a pure quantum process to be calculated. This is especially useful for testing small subcircuits that will be used many times in a larger computation. # @@ -183,7 +183,7 @@ unitary = result.get_unitary() print(unitary.round(1).real) -# ## `pytket.extensions.pyquil.ForestBackend` +# ## `ForestBackend` # Whilst it can, with suitable credentials, be used to access the Rigetti QPUs, the `ForestBackend` also features a simulator mode which turns it into a noiseless sampling simulator that matches the constraints of the simulated device (e.g. the same gate set, restricted connectivity, measurement model, etc.). This is useful when playing around with custom compilation strategies to ensure that your final circuits are suitable to run on the device and for checking that your overall program works fine before you invest in reserving a QPU. # @@ -194,18 +194,18 @@ # `docker run --rm -it -p 5555:5555 rigetti/quilc -R` # `docker run --rm -it -p 5000:5000 rigetti/qvm -S` -# ## `pytket.extensions.pyquil.ForestStateBackend` +# ## `ForestStateBackend` # The Rigetti `pyquil` package also provides the `WavefunctionSimulator`, which we present as the `ForestStateBackend`. Functionally, it is very similar to the `AerStateBackend` so can be used interchangeably. It does require that `quilc` and `qvm` are running as separate processes when not running on a Rigetti QMI. # # Useful features: # - support for fast expectation value calculations according to `QubitPauliString`s or Hermitian `QubitPauliOperator`s. -# ## `pytket.extensions.qsharp.QsharpSimulatorBackend` +# ## `QsharpSimulatorBackend` # The `QsharpSimulatorBackend` is another basic sampling simulator that is interchangeable with others, using the Microsoft QDK simulator. Note that the `pytket-qsharp` package is dependent on the `dotnet` SDK and `iqsharp` tool. Please consult the `pytket-qsharp` installation instructions for recommendations. -# ## `pytket.extensions.qsharp.QsharpToffoliSimulatorBackend` +# ## `QsharpToffoliSimulatorBackend` # Toffoli circuits form a strict fragment of quantum circuits and can be efficiently simulated. The `QsharpToffoliSimulatorBackend` can only operate on these circuits, but scales much better with system size than regular simulators. # @@ -232,7 +232,7 @@ counts = backend.get_result(handle).get_counts() print(counts) -# ## `pytket.extensions.qsharp.QsharpEstimatorBackend` +# ## `QsharpEstimatorBackend` # The `QsharpEstimatorBackend` is not strictly a simulator, as it doesn't model the state of the quantum system and try to identify the final state, but instead analyses the circuit to estimate the required resources to run it. It does not support any of the regular outcome types (e.g. shots, counts, statevector), just the summary of the estimated resources. # @@ -261,7 +261,7 @@ # resources = backend.get_resources(handle) # print(resources) -# ## `pytket.extensions.qulacs.QulacsBackend` +# ## `QulacsBackend` # The `QulacsBackend` is an all-purpose simulator with both sampling and statevector modes, using the basic CPU simulator from Qulacs. # @@ -271,14 +271,14 @@ # Useful features: # - support for fast expectation value calculations according to `QubitPauliString`s or Hermitian `QubitPauliOperator`s. -# ## `pytket.extensions.qulacs.QulacsGPUBackend` +# ## `QulacsGPUBackend` # If the GPU version of Qulacs is installed, the `QulacsGPUBackend` will use that to benefit from even faster speeds. It is very easy to get started with using a GPU, as it only requires a CUDA installation and the `qulacs-gpu` package from `pip`. Functionally, it is identical to the `QulacsBackend`, but potentially faster if you have GPU resources available. # # Unique features: # - GPU support for very fast simulation. -# ## `pytket.extensions.projectq.ProjectQBackend` +# ## `ProjectQBackend` # ProjectQ is a popular quantum circuit simulator, thanks to its availability and ease of use. It provides a similar level of performance and features to `AerStateBackend`. # From 5cd76e903b4d8d2592d83149bebfb0bc091738fd Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 26 Oct 2023 17:43:09 +0100 Subject: [PATCH 10/51] impove example on conditional gates --- examples/conditional_gate_example.ipynb | 2 +- examples/python/conditional_gate_example.py | 71 ++++++++++----------- 2 files changed, 35 insertions(+), 38 deletions(-) diff --git a/examples/conditional_gate_example.ipynb b/examples/conditional_gate_example.ipynb index 108d62de..f2ecc9da 100644 --- a/examples/conditional_gate_example.ipynb +++ b/examples/conditional_gate_example.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Conditional Execution"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Whilst any quantum process can be created by performing \"pure\" operations delaying all measurements to the end, this is not always practical and can greatly increase the resource requirements. It is much more convenient to alternate quantum gates and measurements, especially if we can use the measurement results to determine which gates to apply (we refer to this more generic circuit model as \"mixed\" circuits, against the usual \"pure\" circuits). This is especially crucial for error correcting codes, where the correction gates are applied only if an error is detected.
\n", "
\n", "Measurements on many NISQ devices are often slow and it is hard to maintain other qubits in a quantum state during the measurement operation. Hence they may only support a single round of measurements at the end of the circuit, removing the need for conditional gate support. However, the ability to work with mid-circuit measurement and conditional gates is a feature in high demand for the future, and tket is ready for it.
\n", "
\n", "Not every circuit language specification supports conditional gates in the same way. The most popular circuit model at the moment is that provided by the OpenQASM language. This permits a very restricted model of classical logic, where we can apply a gate conditionally on the exact value of a classical register. There is no facility in the current spec for Boolean logic or classical operations to apply any function to the value prior to the equality check.
\n", "
\n", "For example, quantum teleportation can be performed by the following QASM:
\n", "`OPENQASM 2.0;`
\n", "`include \"qelib1.inc\";`
\n", "`qreg a[2];`
\n", "`qreg b[1];`
\n", "`creg c[2];`
\n", "`// Bell state between Alice and Bob`
\n", "`h a[1];`
\n", "`cx a[1],b[0];`
\n", "`// Bell measurement of Alice's qubits`
\n", "`cx a[0],a[1];`
\n", "`h a[0];`
\n", "`measure a[0] -> c[0];`
\n", "`measure a[1] -> c[1];`
\n", "`// Correction of Bob's qubit`
\n", "`if(c==1) z b[0];`
\n", "`if(c==3) z b[0];`
\n", "`if(c==2) x b[0];`
\n", "`if(c==3) x b[0];`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["tket supports a slightly more general form of conditional gates, where the gate is applied conditionally on the exact value of any list of bits. When adding a gate to a `Circuit` object, pass in the kwargs `condition_bits` and `condition_value` and the gate will only be applied if the state of the bits yields the binary representation of the value."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit()\n", "alice = c.add_q_register(\"a\", 2)\n", "bob = c.add_q_register(\"b\", 1)\n", "cr = c.add_c_register(\"c\", 2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Bell state between Alice and Bob:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.H(alice[1])\n", "c.CX(alice[1], bob[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Bell measurement of Alice's qubits:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.CX(alice[0], alice[1])\n", "c.H(alice[0])\n", "c.Measure(alice[0], cr[0])\n", "c.Measure(alice[1], cr[1])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Correction of Bob's qubit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.Z(bob[0], condition_bits=[cr[0]], condition_value=1)\n", "c.X(bob[0], condition_bits=[cr[1]], condition_value=1)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Performing individual gates conditionally is sufficient, but can get cumbersome for larger circuits. Fortunately, tket's Box structures can also be performed conditionally, enabling this to be applied to large circuits with ease.
\n", "
\n", "For the sake of example, assume our device struggles to perform $X$ gates. We can surround it by $CX$ gates onto an ancilla, so measuring the ancilla will either result in the identity or $X$ being applied to the target qubit. If we detect that the $X$ fails, we can retry."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import CircBox, Qubit, Bit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["checked_x = Circuit(2, 1)\n", "checked_x.CX(0, 1)\n", "checked_x.X(0)\n", "checked_x.CX(0, 1)\n", "checked_x.Measure(1, 0)\n", "x_box = CircBox(checked_x)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit()\n", "target = Qubit(\"t\", 0)\n", "ancilla = Qubit(\"a\", 0)\n", "success = Bit(\"s\", 0)\n", "c.add_qubit(target)\n", "c.add_qubit(ancilla)\n", "c.add_bit(success)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Try the X gate:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.add_circbox(x_box, args=[target, ancilla, success])\n", "# Try again if the X failed\n", "c.add_circbox(\n", " x_box, args=[target, ancilla, success], condition_bits=[success], condition_value=0\n", ")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["tket is able to apply essential compilation passes on circuits containing conditional gates. This includes decomposing any boxes into primitive gates and rebasing to other gatesets whilst preserving the conditional data."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.passes import DecomposeBoxes, RebaseTket, SequencePass"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["comp_pass = SequencePass([DecomposeBoxes(), RebaseTket()])"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["comp_pass.apply(c)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c"]}, {"cell_type": "markdown", "metadata": {}, "source": ["A tket circuit can be converted to OpenQASM or other languages following the same classical model (e.g. Qiskit) when all conditional gates are dependent on the exact state of a single, whole classical register."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import tk_to_qiskit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qc = tk_to_qiskit(c)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(qc)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This allows us to test our mixed programs using the `AerBackend`."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import AerBackend"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(2, 1)\n", "c.Rx(0.3, 0)\n", "c.Measure(0, 0)\n", "# Set qubit 1 to be the opposite result and measure\n", "c.X(1, condition_bits=[0], condition_value=0)\n", "c.Measure(1, 0)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()\n", "c = backend.get_compiled_circuit(c)\n", "counts = backend.run_circuit(c, 1024).get_counts()\n", "print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Try out mid-circuit measurement and conditional gate support on the `AerBackend` simulator, or ask about accessing the `QuantinuumBackend` to try on a hardware device."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Conditional Gates"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Whilst any quantum process can be created by performing \"pure\" operations delaying all measurements to the end, this is not always practical and can greatly increase the resource requirements. It is much more convenient to alternate quantum gates and measurements, especially if we can use the measurement results to determine which gates to apply (we refer to this more generic circuit model as \"mixed\" circuits, against the usual \"pure\" circuits). This is especially crucial for error correcting codes, where the correction gates are applied only if an error is detected.
\n", "
\n", "Measurements on many NISQ devices are often slow and it is hard to maintain other qubits in a quantum state during the measurement operation. Hence they may only support a single round of measurements at the end of the circuit, removing the need for conditional gate support. However, the ability to work with mid-circuit measurement and conditional gates is a feature in high demand for the future, and tket is ready for it.
\n", "
\n", "Not every circuit language specification supports conditional gates in the same way. The most popular circuit model at the moment is that provided by the OpenQASM language. This permits a very restricted model of classical logic, where we can apply a gate conditionally on the exact value of a classical register. There is no facility in the current spec for Boolean logic or classical operations to apply any function to the value prior to the equality check.
\n", "
\n", "For example, quantum teleportation can be performed by the following QASM:
\n", "`OPENQASM 2.0;`
\n", "`include \"qelib1.inc\";`
\n", "`qreg a[2];`
\n", "`qreg b[1];`
\n", "`creg c[2];`
\n", "`// Bell state between Alice and Bob`
\n", "`h a[1];`
\n", "`cx a[1],b[0];`
\n", "`// Bell measurement of Alice's qubits`
\n", "`cx a[0],a[1];`
\n", "`h a[0];`
\n", "`measure a[0] -> c[0];`
\n", "`measure a[1] -> c[1];`
\n", "`// Correction of Bob's qubit`
\n", "`if(c==1) z b[0];`
\n", "`if(c==3) z b[0];`
\n", "`if(c==2) x b[0];`
\n", "`if(c==3) x b[0];`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["tket supports a slightly more general form of conditional gates, where the gate is applied conditionally on the exact value of any list of bits. When adding a gate to a `Circuit` object, pass in the kwargs `condition_bits` and `condition_value` and the gate will only be applied if the state of the bits yields the binary representation of the value."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.circuit.display import render_circuit_jupyter"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ = Circuit()\n", "alice = circ.add_q_register(\"a\", 2)\n", "bob = circ.add_q_register(\"b\", 1)\n", "cr = circ.add_c_register(\"c\", 2)\n", "# Bell state between Alice and Bob:\n", "circ.H(alice[1])\n", "circ.CX(alice[1], bob[0])\n", "# Bell measurement of Alice's qubits:\n", "circ.CX(alice[0], alice[1])\n", "circ.H(alice[0])\n", "circ.Measure(alice[0], cr[0])\n", "circ.Measure(alice[1], cr[1])\n", "# Correction of Bob's qubit:\n", "circ.Z(bob[0], condition_bits=[cr[0]], condition_value=1)\n", "circ.X(bob[0], condition_bits=[cr[1]], condition_value=1)\n", "render_circuit_jupyter(circ)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Performing individual gates conditionally is sufficient, but can get cumbersome for larger circuits. Fortunately, tket's Box structures can also be performed conditionally, enabling this to be applied to large circuits with ease.
\n", "
\n", "For the sake of example, assume our device struggles to perform $X$ gates. We can surround it by $CX$ gates onto an ancilla, so measuring the ancilla will either result in the identity or $X$ being applied to the target qubit. If we detect that the $X$ fails, we can retry."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import CircBox, Qubit, Bit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["checked_x = Circuit(2, 1)\n", "checked_x.CX(0, 1)\n", "checked_x.X(0)\n", "checked_x.CX(0, 1)\n", "checked_x.Measure(1, 0)\n", "x_box = CircBox(checked_x)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ2 = Circuit()\n", "target = Qubit(\"t\", 0)\n", "ancilla = Qubit(\"a\", 0)\n", "success = Bit(\"s\", 0)\n", "circ2.add_qubit(target)\n", "circ2.add_qubit(ancilla)\n", "circ2.add_bit(success)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Try the X gate:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ2.add_circbox(x_box, args=[target, ancilla, success])\n", "# Try again if the X failed\n", "circ2.add_circbox(\n", " x_box, args=[target, ancilla, success], condition_bits=[success], condition_value=0\n", ")\n", "render_circuit_jupyter(circ2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["tket is able to apply essential compilation passes on circuits containing conditional gates. This includes decomposing any boxes into primitive gates and rebasing to other gatesets whilst preserving the conditional data."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.passes import DecomposeBoxes, RebaseTket, SequencePass"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["comp_pass = SequencePass([DecomposeBoxes(), RebaseTket()])\n", "comp_pass.apply(circ2)\n", "render_circuit_jupyter(circ2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["A tket circuit can be converted to OpenQASM or other languages following the same classical model (e.g. Qiskit) when all conditional gates are dependent on the exact state of a single, whole classical register."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import tk_to_qiskit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qc = tk_to_qiskit(circ2)\n", "print(qc)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This allows us to test our mixed programs using the `AerBackend`."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import AerBackend"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ3 = Circuit(2, 1)\n", "circ3.Rx(0.3, 0)\n", "circ3.Measure(0, 0)\n", "# Set qubit 1 to be the opposite result and measure\n", "circ3.X(1, condition_bits=[0], condition_value=0)\n", "circ3.Measure(1, 0)\n", "backend = AerBackend()\n", "compiled_circ = backend.get_compiled_circuit(circ3)\n", "render_circuit_jupyter(compiled_circ)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["counts = backend.run_circuit(compiled_circ, 1024).get_counts()\n", "print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Try out mid-circuit measurement and conditional gate support on the `AerBackend` simulator, or ask about accessing the `QuantinuumBackend` to try on a hardware device."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file diff --git a/examples/python/conditional_gate_example.py b/examples/python/conditional_gate_example.py index 1e268f5b..81bb8110 100644 --- a/examples/python/conditional_gate_example.py +++ b/examples/python/conditional_gate_example.py @@ -1,4 +1,4 @@ -# # Conditional Execution +# # Conditional Gates # Whilst any quantum process can be created by performing "pure" operations delaying all measurements to the end, this is not always practical and can greatly increase the resource requirements. It is much more convenient to alternate quantum gates and measurements, especially if we can use the measurement results to determine which gates to apply (we refer to this more generic circuit model as "mixed" circuits, against the usual "pure" circuits). This is especially crucial for error correcting codes, where the correction gates are applied only if an error is detected. # @@ -29,28 +29,25 @@ # tket supports a slightly more general form of conditional gates, where the gate is applied conditionally on the exact value of any list of bits. When adding a gate to a `Circuit` object, pass in the kwargs `condition_bits` and `condition_value` and the gate will only be applied if the state of the bits yields the binary representation of the value. from pytket import Circuit +from pytket.circuit.display import render_circuit_jupyter -c = Circuit() -alice = c.add_q_register("a", 2) -bob = c.add_q_register("b", 1) -cr = c.add_c_register("c", 2) - +circ = Circuit() +alice = circ.add_q_register("a", 2) +bob = circ.add_q_register("b", 1) +cr = circ.add_c_register("c", 2) # Bell state between Alice and Bob: - -c.H(alice[1]) -c.CX(alice[1], bob[0]) - +circ.H(alice[1]) +circ.CX(alice[1], bob[0]) # Bell measurement of Alice's qubits: - -c.CX(alice[0], alice[1]) -c.H(alice[0]) -c.Measure(alice[0], cr[0]) -c.Measure(alice[1], cr[1]) - +circ.CX(alice[0], alice[1]) +circ.H(alice[0]) +circ.Measure(alice[0], cr[0]) +circ.Measure(alice[1], cr[1]) # Correction of Bob's qubit: +circ.Z(bob[0], condition_bits=[cr[0]], condition_value=1) +circ.X(bob[0], condition_bits=[cr[1]], condition_value=1) +render_circuit_jupyter(circ) -c.Z(bob[0], condition_bits=[cr[0]], condition_value=1) -c.X(bob[0], condition_bits=[cr[1]], condition_value=1) # Performing individual gates conditionally is sufficient, but can get cumbersome for larger circuits. Fortunately, tket's Box structures can also be performed conditionally, enabling this to be applied to large circuits with ease. # @@ -65,54 +62,54 @@ checked_x.Measure(1, 0) x_box = CircBox(checked_x) -c = Circuit() +circ2 = Circuit() target = Qubit("t", 0) ancilla = Qubit("a", 0) success = Bit("s", 0) -c.add_qubit(target) -c.add_qubit(ancilla) -c.add_bit(success) +circ2.add_qubit(target) +circ2.add_qubit(ancilla) +circ2.add_bit(success) # Try the X gate: -c.add_circbox(x_box, args=[target, ancilla, success]) +circ2.add_circbox(x_box, args=[target, ancilla, success]) # Try again if the X failed -c.add_circbox( +circ2.add_circbox( x_box, args=[target, ancilla, success], condition_bits=[success], condition_value=0 ) +render_circuit_jupyter(circ2) # tket is able to apply essential compilation passes on circuits containing conditional gates. This includes decomposing any boxes into primitive gates and rebasing to other gatesets whilst preserving the conditional data. from pytket.passes import DecomposeBoxes, RebaseTket, SequencePass comp_pass = SequencePass([DecomposeBoxes(), RebaseTket()]) +comp_pass.apply(circ2) +render_circuit_jupyter(circ2) -comp_pass.apply(c) - -c # A tket circuit can be converted to OpenQASM or other languages following the same classical model (e.g. Qiskit) when all conditional gates are dependent on the exact state of a single, whole classical register. from pytket.extensions.qiskit import tk_to_qiskit -qc = tk_to_qiskit(c) - +qc = tk_to_qiskit(circ2) print(qc) # This allows us to test our mixed programs using the `AerBackend`. from pytket.extensions.qiskit import AerBackend -c = Circuit(2, 1) -c.Rx(0.3, 0) -c.Measure(0, 0) +circ3 = Circuit(2, 1) +circ3.Rx(0.3, 0) +circ3.Measure(0, 0) # Set qubit 1 to be the opposite result and measure -c.X(1, condition_bits=[0], condition_value=0) -c.Measure(1, 0) - +circ3.X(1, condition_bits=[0], condition_value=0) +circ3.Measure(1, 0) backend = AerBackend() -c = backend.get_compiled_circuit(c) -counts = backend.run_circuit(c, 1024).get_counts() +compiled_circ = backend.get_compiled_circuit(circ3) +render_circuit_jupyter(compiled_circ) + +counts = backend.run_circuit(compiled_circ, 1024).get_counts() print(counts) # Try out mid-circuit measurement and conditional gate support on the `AerBackend` simulator, or ask about accessing the `QuantinuumBackend` to try on a hardware device. From 858a408db3c7f88001b3d5ef90bc2b5384302b5c Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 26 Oct 2023 17:44:35 +0100 Subject: [PATCH 11/51] clean up table of contents --- examples/_toc.yml | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/examples/_toc.yml b/examples/_toc.yml index be8f2680..13955abd 100644 --- a/examples/_toc.yml +++ b/examples/_toc.yml @@ -6,9 +6,9 @@ root: README parts: - caption: Building Quantum Circuits chapters: + - file: circuit_generation_example - file: circuit_analysis_example - file: conditional_gate_example - - file: circuit_generation_example - caption: TKET backends chapters: - file: backends_example @@ -18,11 +18,12 @@ parts: - file: qiskit_integration - caption: Circuit compilation chapters: + - file: compilation_example + - file: symbolics_example - file: mapping_example - file: measurement_reduction_example - file: contextual_optimization - file: symbolics_example - - file: compilation_example - file: ansatz_sequence_example - caption: Algorithm Demos chapters: From 391317555953b13f516f30f5f3ea36c78cc5e9e4 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 26 Oct 2023 17:45:04 +0100 Subject: [PATCH 12/51] change tket -> TKET --- examples/python/entanglement_swapping.py | 2 +- examples/python/ucc_vqe.py | 2 +- examples/tket_benchmarking.ipynb | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/examples/python/entanglement_swapping.py b/examples/python/entanglement_swapping.py index a8e1c49f..1c39beb1 100644 --- a/examples/python/entanglement_swapping.py +++ b/examples/python/entanglement_swapping.py @@ -1,4 +1,4 @@ -# # Iterated Entanglement Swapping using tket +# # Iterated Entanglement Swapping using TKET # In this tutorial, we will focus on: # - designing circuits with mid-circuit measurement and conditional gates; diff --git a/examples/python/ucc_vqe.py b/examples/python/ucc_vqe.py index b38fb7ef..d09500fc 100644 --- a/examples/python/ucc_vqe.py +++ b/examples/python/ucc_vqe.py @@ -1,4 +1,4 @@ -# # VQE for Unitary Coupled Cluster using tket +# # VQE for Unitary Coupled Cluster using TKET # In this tutorial, we will focus on: # - building parameterised ansätze for variational algorithms; diff --git a/examples/tket_benchmarking.ipynb b/examples/tket_benchmarking.ipynb index 0dfe0463..26aae780 100644 --- a/examples/tket_benchmarking.ipynb +++ b/examples/tket_benchmarking.ipynb @@ -4,7 +4,7 @@ "cell_type": "markdown", "metadata": {}, "source": [ - "# tket benchmarking example\n" + "# TKET benchmarking example\n" ] }, { From f9e03ad52e99d2046aa53809ee4c6924178aadec Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 26 Oct 2023 17:46:01 +0100 Subject: [PATCH 13/51] remove duplicate file name --- examples/_toc.yml | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/examples/_toc.yml b/examples/_toc.yml index 13955abd..efa98413 100644 --- a/examples/_toc.yml +++ b/examples/_toc.yml @@ -21,10 +21,9 @@ parts: - file: compilation_example - file: symbolics_example - file: mapping_example + - file: ansatz_sequence_example - file: measurement_reduction_example - file: contextual_optimization - - file: symbolics_example - - file: ansatz_sequence_example - caption: Algorithm Demos chapters: - file: ucc_vqe From 2d0efff143893eb8dea60efec980d4cac19e0da9 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 26 Oct 2023 17:46:20 +0100 Subject: [PATCH 14/51] change headings in benchmarking and VQE notebooks --- examples/entanglement_swapping.ipynb | 2 +- examples/ucc_vqe.ipynb | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/examples/entanglement_swapping.ipynb b/examples/entanglement_swapping.ipynb index 6592a721..8b1f5137 100644 --- a/examples/entanglement_swapping.ipynb +++ b/examples/entanglement_swapping.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Iterated Entanglement Swapping using tket"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In this tutorial, we will focus on:
\n", "- designing circuits with mid-circuit measurement and conditional gates;
\n", "- utilising noise models in supported simulators."]}, {"cell_type": "markdown", "metadata": {}, "source": ["This example assumes the reader is familiar with the Qubit Teleportation and Entanglement Swapping protocols, and basic models of noise in quantum devices.
\n", "
\n", "To run this example, you will need `pytket`, `pytket-qiskit`, and `plotly` (installed via `pip`). To view the graphs, you will need an intallation of `plotly-orca`.
\n", "
\n", "Current quantum hardware fits into the NISQ (Noisy, Intermediate-Scale Quantum) regime. This noise cannot realistically be combatted using conventional error correcting codes, because of the lack of available qubits, noise levels exceeding the code thresholds, and very few devices available that can perform measurements and corrections mid-circuit. Analysis of how quantum algorithms perform under noisy conditions is a very active research area, as is finding ways to cope with it. Here, we will look at how well we can perform the Entanglement Swapping protocol with different noise levels.
\n", "
\n", "The Entanglement Swapping protocol requires two parties to share Bell pairs with a third party, who applies the Qubit Teleportation protocol to generate a Bell pair between the two parties. The Qubit Teleportation step requires us to be able to measure some qubits and make subsequent corrections to the remaining qubits. There are only a handful of simulators and devices that currently support this, with others restricted to only measuring the qubits at the end of the circuit.
\n", "
\n", "The most popular circuit model with conditional gates at the moment is that provided by the OpenQASM language. This permits a very restricted model of classical logic, where we can apply a gate conditionally on the exact value of a classical register. There is no facility in the current spec for Boolean logic or classical operations to apply any function to the value prior to the equality check. For example, Qubit Teleportation can be performed by the following QASM:
\n", "`OPENQASM 2.0;`
\n", "`include \"qelib1.inc\";`
\n", "`qreg a[2];`
\n", "`qreg b[1];`
\n", "`creg c[2];`
\n", "`// Bell state between Alice and Bob`
\n", "`h a[1];`
\n", "`cx a[1],b[0];`
\n", "`// Bell measurement of Alice's qubits`
\n", "`cx a[0],a[1];`
\n", "`h a[0];`
\n", "`measure a[0] -> c[0];`
\n", "`measure a[1] -> c[1];`
\n", "`// Correction of Bob's qubit`
\n", "`if(c==1) z b[0];`
\n", "`if(c==3) z b[0];`
\n", "`if(c==2) x b[0];`
\n", "`if(c==3) x b[0];`
\n", "
\n", "This corresponds to the following `pytket` code:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qtel = Circuit()\n", "alice = qtel.add_q_register(\"a\", 2)\n", "bob = qtel.add_q_register(\"b\", 1)\n", "data = qtel.add_c_register(\"d\", 2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Bell state between Alice and Bob:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qtel.H(alice[1])\n", "qtel.CX(alice[1], bob[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Bell measurement of Alice's qubits:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qtel.CX(alice[0], alice[1])\n", "qtel.H(alice[0])\n", "qtel.Measure(alice[0], data[0])\n", "qtel.Measure(alice[1], data[1])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Correction of Bob's qubit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qtel.X(bob[0], condition_bits=[data[0], data[1]], condition_value=2)\n", "qtel.X(bob[0], condition_bits=[data[0], data[1]], condition_value=3)\n", "qtel.Z(bob[0], condition_bits=[data[0], data[1]], condition_value=1)\n", "qtel.Z(bob[0], condition_bits=[data[0], data[1]], condition_value=3)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["So to demonstrate the Entanglement Swapping protocol, we just need to run this on one side of a Bell pair."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["es = Circuit()\n", "ava = es.add_q_register(\"a\", 1)\n", "bella = es.add_q_register(\"b\", 2)\n", "charlie = es.add_q_register(\"c\", 1)\n", "data = es.add_c_register(\"d\", 2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Bell state between Ava and Bella:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["es.H(ava[0])\n", "es.CX(ava[0], bella[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Teleport `bella[0]` to `charlie[0]`:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["tel_to_c = qtel.copy()\n", "tel_to_c.rename_units({alice[0]: bella[0], alice[1]: bella[1], bob[0]: charlie[0]})\n", "es.append(tel_to_c)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(es.get_commands())"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Let's start by running a noiseless simulation of this to verify that what we get looks like a Bell pair."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import AerBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Connect to a simulator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Make a ZZ measurement of the Bell pair:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["bell_test = es.copy()\n", "bell_test.Measure(ava[0], data[0])\n", "bell_test.Measure(charlie[0], data[1])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run the experiment:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["bell_test = backend.get_compiled_circuit(bell_test)\n", "from pytket.circuit.display import render_circuit_jupyter"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["render_circuit_jupyter(bell_test)\n", "handle = backend.process_circuit(bell_test, n_shots=2000)\n", "counts = backend.get_result(handle).get_counts()"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is good, we have got roughly 50/50 measurement results of 00 and 11 under the ZZ operator. But there are many other states beyond the Bell state that also generate this distribution, so to gain more confidence in our claim about the state we should make more measurements that also characterise it, i.e. perform state tomography.
\n", "
\n", "Here, we will demonstrate a naive approach to tomography that makes 3^n measurement circuits for an n-qubit state. More elaborate methods also exist."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.utils import append_pauli_measurement, probs_from_counts\n", "from itertools import product\n", "from scipy.linalg import lstsq, eigh\n", "import numpy as np"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def gen_tomography_circuits(state, qubits, bits):\n", " # Yields {X, Y, Z}^n measurements in lexicographical order\n", " # Only measures qubits, storing the result in bits\n", " # (since we don't care about the ancilla qubits)\n", " assert len(qubits) == len(bits)\n", " for paulis in product([Pauli.X, Pauli.Y, Pauli.Z], repeat=len(qubits)):\n", " circ = state.copy()\n", " for qb, b, p in zip(qubits, bits, paulis):\n", " if p == Pauli.X:\n", " circ.H(qb)\n", " elif p == Pauli.Y:\n", " circ.V(qb)\n", " circ.Measure(qb, b)\n", " yield circ"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def run_tomography_circuits(state, qubits, bits, backend):\n", " circs = list(gen_tomography_circuits(state, qubits, bits))\n", " # Compile and run each circuit\n", " circs = backend.get_compiled_circuits(circs)\n", " handles = backend.process_circuits(circs, n_shots=2000)\n", " # Get the observed measurement probabilities\n", " probs_list = []\n", " for result in backend.get_results(handles):\n", " counts = result.get_counts()\n", " probs = probs_from_counts(counts)\n", " probs_list.append(probs)\n", " return probs_list"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def fit_tomography_outcomes(probs_list, n_qbs):\n", " # Define the density matrices for the basis states\n", " basis = dict()\n", " basis[(Pauli.X, 0)] = np.asarray([[0.5, 0.5], [0.5, 0.5]])\n", " basis[(Pauli.X, 1)] = np.asarray([[0.5, -0.5], [-0.5, 0.5]])\n", " basis[(Pauli.Y, 0)] = np.asarray([[0.5, -0.5j], [0.5j, 0.5]])\n", " basis[(Pauli.Y, 1)] = np.asarray([[0.5, 0.5j], [-0.5j, 0.5]])\n", " basis[(Pauli.Z, 0)] = np.asarray([[1, 0], [0, 0]])\n", " basis[(Pauli.Z, 1)] = np.asarray([[0, 0], [0, 1]])\n", " dim = 2**n_qbs\n", " # Define vector all_probs as a concatenation of probability vectors for each measurement (2**n x 3**n, 1)\n", " # Define matrix all_ops mapping a (vectorised) density matrix to a vector of probabilities for each measurement\n", " # (2**n x 3**n, 2**n x 2**n)\n", " all_probs = []\n", " all_ops = []\n", " for paulis, probs in zip(\n", " product([Pauli.X, Pauli.Y, Pauli.Z], repeat=n_qbs), probs_list\n", " ):\n", " prob_vec = []\n", " meas_ops = []\n", " for outcome in product([0, 1], repeat=n_qbs):\n", " prob_vec.append(probs.get(outcome, 0))\n", " op = np.eye(1, dtype=complex)\n", " for p, o in zip(paulis, outcome):\n", " op = np.kron(op, basis[(p, o)])\n", " meas_ops.append(op.reshape(1, dim * dim).conj())\n", " all_probs.append(np.vstack(prob_vec))\n", " all_ops.append(np.vstack(meas_ops))\n", " # Solve for density matrix by minimising || all_ops * dm - all_probs ||\n", " dm, _, _, _ = lstsq(np.vstack(all_ops), np.vstack(all_probs))\n", " dm = dm.reshape(dim, dim)\n", " # Make density matrix positive semi-definite\n", " v, w = eigh(dm)\n", " for i in range(dim):\n", " if v[i] < 0:\n", " for j in range(i + 1, dim):\n", " v[j] += v[i] / (dim - (i + 1))\n", " v[i] = 0\n", " dm = np.zeros([dim, dim], dtype=complex)\n", " for j in range(dim):\n", " dm += v[j] * np.outer(w[:, j], np.conj(w[:, j]))\n", " # Normalise trace of density matrix\n", " dm /= np.trace(dm)\n", " return dm"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["probs_list = run_tomography_circuits(\n", " es, [ava[0], charlie[0]], [data[0], data[1]], backend\n", ")\n", "dm = fit_tomography_outcomes(probs_list, 2)\n", "print(dm.round(3))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is very close to the true density matrix for a pure Bell state. We can attribute the error here to the sampling error since we only take 2000 samples of each measurement circuit.
\n", "
\n", "To quantify exactly how similar it is to the correct density matrix, we can calculate the fidelity."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from scipy.linalg import sqrtm"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def fidelity(dm0, dm1):\n", " # Calculate the fidelity between two density matrices\n", " sq0 = sqrtm(dm0)\n", " sq1 = sqrtm(dm1)\n", " return np.linalg.norm(sq0.dot(sq1)) ** 2"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["bell_state = np.asarray(\n", " [\n", " [0.5, 0, 0, 0.5],\n", " [0, 0, 0, 0],\n", " [0, 0, 0, 0],\n", " [0.5, 0, 0, 0.5],\n", " ]\n", ")\n", "print(fidelity(dm, bell_state))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This high fidelity is unsurprising since we have a completely noiseless simulation. So the next step is to add some noise to the simulation and observe how the overall fidelity is affected. The `AerBackend` wraps around the Qiskit Aer simulator and can pass on any `qiskit.providers.aer.noise.NoiseModel` to the simulator. Let's start by adding some uniform depolarising noise to each CX gate and some uniform measurement error."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from qiskit.providers.aer.noise import NoiseModel, depolarizing_error, ReadoutError"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def make_noise_model(dep_err_rate, ro_err_rate, qubits):\n", " # Define a noise model that applies uniformly to the given qubits\n", " model = NoiseModel()\n", " dep_err = depolarizing_error(dep_err_rate, 2)\n", " ro_err = ReadoutError(\n", " [[1 - ro_err_rate, ro_err_rate], [ro_err_rate, 1 - ro_err_rate]]\n", " )\n", " # Add depolarising error to CX gates between any qubits (implying full connectivity)\n", " for i, j in product(qubits, repeat=2):\n", " if i != j:\n", " model.add_quantum_error(dep_err, [\"cx\"], [i, j])\n", " # Add readout error for each qubit\n", " for i in qubits:\n", " model.add_readout_error(ro_err, qubits=[i])\n", " return model"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_model = make_noise_model(0.03, 0.05, range(4))\n", "backend = AerBackend(noise_model=test_model)\n", "probs_list = run_tomography_circuits(\n", " es, [ava[0], charlie[0]], [data[0], data[1]], backend\n", ")\n", "dm = fit_tomography_outcomes(probs_list, 2)\n", "print(dm.round(3))\n", "print(fidelity(dm, bell_state))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Despite the very small circuit and the relatively small error rates, the fidelity of the final state has reduced considerably.
\n", "
\n", "As far as circuits go, the entanglement swapping protocol is little more than a toy example and is nothing close to the scale of circuits for most interesting quantum computational problems. However, it is possible to iterate the protocol many times to build up a larger computation, allowing us to see the impact of the noise at different scales."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import OpType\n", "from plotly.graph_objects import Scatter, Figure"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def iterated_entanglement_swap(n_iter):\n", " # Iterate the entanglement swapping protocol n_iter times\n", " it_es = Circuit()\n", " ava = it_es.add_q_register(\"a\", 1)\n", " bella = it_es.add_q_register(\"b\", 2)\n", " charlie = it_es.add_q_register(\"c\", 1)\n", " data = it_es.add_c_register(\"d\", 2)\n\n", " # Start with an initial Bell state\n", " it_es.H(ava[0])\n", " it_es.CX(ava[0], bella[0])\n", " for i in range(n_iter):\n", " if i % 2 == 0:\n", " # Teleport bella[0] to charlie[0] to give a Bell pair between ava[0] and charlier[0]\n", " tel_to_c = qtel.copy()\n", " tel_to_c.rename_units(\n", " {alice[0]: bella[0], alice[1]: bella[1], bob[0]: charlie[0]}\n", " )\n", " it_es.append(tel_to_c)\n", " it_es.add_gate(OpType.Reset, [bella[0]])\n", " it_es.add_gate(OpType.Reset, [bella[1]])\n", " else:\n", " # Teleport charlie[0] to bella[0] to give a Bell pair between ava[0] and bella[0]\n", " tel_to_b = qtel.copy()\n", " tel_to_b.rename_units(\n", " {alice[0]: charlie[0], alice[1]: bella[1], bob[0]: bella[0]}\n", " )\n", " it_es.append(tel_to_b)\n", " it_es.add_gate(OpType.Reset, [bella[1]])\n", " it_es.add_gate(OpType.Reset, [charlie[0]])\n", " # Return the circuit and the qubits expected to share a Bell pair\n", " if n_iter % 2 == 0:\n", " return it_es, [ava[0], bella[0]]\n", " else:\n", " return it_es, [ava[0], charlie[0]]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def iterated_noisy_experiment(dep_err_rate, ro_err_rate, max_iter):\n", " # Set up the noisy simulator with the given error rates\n", " test_model = make_noise_model(dep_err_rate, ro_err_rate, range(4))\n", " backend = AerBackend(noise_model=test_model)\n", " # Estimate the fidelity after n iterations, from 0 to max_iter (inclusive)\n", " fid_list = []\n", " for i in range(max_iter + 1):\n", " it_es, qubits = iterated_entanglement_swap(i)\n", " probs_list = run_tomography_circuits(it_es, qubits, [data[0], data[1]], backend)\n", " dm = fit_tomography_outcomes(probs_list, 2)\n", " fid = fidelity(dm, bell_state)\n", " fid_list.append(fid)\n", " return fid_list"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["fig = Figure()\n", "fig.update_layout(\n", " title=\"Iterated Entanglement Swapping under Noise (dep_err = 0.03)\",\n", " xaxis_title=\"Iterations\",\n", " xaxis=dict(range=[0, 10]),\n", " yaxis_title=\"Fidelity\",\n", ")\n", "iter_range = np.arange(11)\n", "for i in range(7):\n", " fids = iterated_noisy_experiment(0.03, 0.025 * i, 10)\n", " plot_data = Scatter(\n", " x=iter_range, y=fids, name=\"ro_err=\" + str(np.round(0.025 * i, 3))\n", " )\n", " fig.add_trace(plot_data)\n", "try:\n", " fig.show(renderer=\"svg\")\n", "except ValueError as e:\n", " print(e) # requires plotly-orca"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["fig = Figure()\n", "fig.update_layout(\n", " title=\"Iterated Entanglement Swapping under Noise (ro_err = 0.05)\",\n", " xaxis_title=\"Iterations\",\n", " xaxis=dict(range=[0, 10]),\n", " yaxis_title=\"Fidelity\",\n", ")\n", "iter_range = np.arange(11)\n", "for i in range(9):\n", " fids = iterated_noisy_experiment(0.01 * i, 0.05, 10)\n", " plot_data = Scatter(\n", " x=iter_range, y=fids, name=\"dep_err=\" + str(np.round(0.01 * i, 3))\n", " )\n", " fig.add_trace(plot_data)\n", "try:\n", " fig.show(renderer=\"svg\")\n", "except ValueError as e:\n", " print(e) # requires plotly-orca"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These graphs are not very surprising, but are still important for seeing that the current error rates of typical NISQ devices become crippling for fidelities very quickly after repeated mid-circuit measurements and corrections (even with this overly-simplified model with uniform noise and no crosstalk or higher error modes). This provides good motivation for the adoption of error mitigation techniques, and for the development of new techniques that are robust to errors in mid-circuit measurements."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Exercises:
\n", "- Vary the fixed noise levels to compare how impactful the depolarising and measurement errors are.
\n", "- Add extra noise characteristics to the noise model to obtain something that more resembles a real device. Possible options include adding error during the reset operations, extending the errors to be non-local, or constructing the noise model from a device's calibration data.
\n", "- Change the circuit from iterated entanglement swapping to iterated applications of a correction circuit from a simple error-correcting code. Do you expect this to be more sensitive to depolarising errors from unitary gates or measurement errors?"]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Iterated Entanglement Swapping using TKET"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In this tutorial, we will focus on:
\n", "- designing circuits with mid-circuit measurement and conditional gates;
\n", "- utilising noise models in supported simulators."]}, {"cell_type": "markdown", "metadata": {}, "source": ["This example assumes the reader is familiar with the Qubit Teleportation and Entanglement Swapping protocols, and basic models of noise in quantum devices.
\n", "
\n", "To run this example, you will need `pytket`, `pytket-qiskit`, and `plotly` (installed via `pip`). To view the graphs, you will need an intallation of `plotly-orca`.
\n", "
\n", "Current quantum hardware fits into the NISQ (Noisy, Intermediate-Scale Quantum) regime. This noise cannot realistically be combatted using conventional error correcting codes, because of the lack of available qubits, noise levels exceeding the code thresholds, and very few devices available that can perform measurements and corrections mid-circuit. Analysis of how quantum algorithms perform under noisy conditions is a very active research area, as is finding ways to cope with it. Here, we will look at how well we can perform the Entanglement Swapping protocol with different noise levels.
\n", "
\n", "The Entanglement Swapping protocol requires two parties to share Bell pairs with a third party, who applies the Qubit Teleportation protocol to generate a Bell pair between the two parties. The Qubit Teleportation step requires us to be able to measure some qubits and make subsequent corrections to the remaining qubits. There are only a handful of simulators and devices that currently support this, with others restricted to only measuring the qubits at the end of the circuit.
\n", "
\n", "The most popular circuit model with conditional gates at the moment is that provided by the OpenQASM language. This permits a very restricted model of classical logic, where we can apply a gate conditionally on the exact value of a classical register. There is no facility in the current spec for Boolean logic or classical operations to apply any function to the value prior to the equality check. For example, Qubit Teleportation can be performed by the following QASM:
\n", "`OPENQASM 2.0;`
\n", "`include \"qelib1.inc\";`
\n", "`qreg a[2];`
\n", "`qreg b[1];`
\n", "`creg c[2];`
\n", "`// Bell state between Alice and Bob`
\n", "`h a[1];`
\n", "`cx a[1],b[0];`
\n", "`// Bell measurement of Alice's qubits`
\n", "`cx a[0],a[1];`
\n", "`h a[0];`
\n", "`measure a[0] -> c[0];`
\n", "`measure a[1] -> c[1];`
\n", "`// Correction of Bob's qubit`
\n", "`if(c==1) z b[0];`
\n", "`if(c==3) z b[0];`
\n", "`if(c==2) x b[0];`
\n", "`if(c==3) x b[0];`
\n", "
\n", "This corresponds to the following `pytket` code:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qtel = Circuit()\n", "alice = qtel.add_q_register(\"a\", 2)\n", "bob = qtel.add_q_register(\"b\", 1)\n", "data = qtel.add_c_register(\"d\", 2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Bell state between Alice and Bob:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qtel.H(alice[1])\n", "qtel.CX(alice[1], bob[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Bell measurement of Alice's qubits:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qtel.CX(alice[0], alice[1])\n", "qtel.H(alice[0])\n", "qtel.Measure(alice[0], data[0])\n", "qtel.Measure(alice[1], data[1])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Correction of Bob's qubit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qtel.X(bob[0], condition_bits=[data[0], data[1]], condition_value=2)\n", "qtel.X(bob[0], condition_bits=[data[0], data[1]], condition_value=3)\n", "qtel.Z(bob[0], condition_bits=[data[0], data[1]], condition_value=1)\n", "qtel.Z(bob[0], condition_bits=[data[0], data[1]], condition_value=3)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["So to demonstrate the Entanglement Swapping protocol, we just need to run this on one side of a Bell pair."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["es = Circuit()\n", "ava = es.add_q_register(\"a\", 1)\n", "bella = es.add_q_register(\"b\", 2)\n", "charlie = es.add_q_register(\"c\", 1)\n", "data = es.add_c_register(\"d\", 2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Bell state between Ava and Bella:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["es.H(ava[0])\n", "es.CX(ava[0], bella[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Teleport `bella[0]` to `charlie[0]`:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["tel_to_c = qtel.copy()\n", "tel_to_c.rename_units({alice[0]: bella[0], alice[1]: bella[1], bob[0]: charlie[0]})\n", "es.append(tel_to_c)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(es.get_commands())"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Let's start by running a noiseless simulation of this to verify that what we get looks like a Bell pair."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import AerBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Connect to a simulator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Make a ZZ measurement of the Bell pair:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["bell_test = es.copy()\n", "bell_test.Measure(ava[0], data[0])\n", "bell_test.Measure(charlie[0], data[1])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run the experiment:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["bell_test = backend.get_compiled_circuit(bell_test)\n", "from pytket.circuit.display import render_circuit_jupyter"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["render_circuit_jupyter(bell_test)\n", "handle = backend.process_circuit(bell_test, n_shots=2000)\n", "counts = backend.get_result(handle).get_counts()"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is good, we have got roughly 50/50 measurement results of 00 and 11 under the ZZ operator. But there are many other states beyond the Bell state that also generate this distribution, so to gain more confidence in our claim about the state we should make more measurements that also characterise it, i.e. perform state tomography.
\n", "
\n", "Here, we will demonstrate a naive approach to tomography that makes 3^n measurement circuits for an n-qubit state. More elaborate methods also exist."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.utils import append_pauli_measurement, probs_from_counts\n", "from itertools import product\n", "from scipy.linalg import lstsq, eigh\n", "import numpy as np"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def gen_tomography_circuits(state, qubits, bits):\n", " # Yields {X, Y, Z}^n measurements in lexicographical order\n", " # Only measures qubits, storing the result in bits\n", " # (since we don't care about the ancilla qubits)\n", " assert len(qubits) == len(bits)\n", " for paulis in product([Pauli.X, Pauli.Y, Pauli.Z], repeat=len(qubits)):\n", " circ = state.copy()\n", " for qb, b, p in zip(qubits, bits, paulis):\n", " if p == Pauli.X:\n", " circ.H(qb)\n", " elif p == Pauli.Y:\n", " circ.V(qb)\n", " circ.Measure(qb, b)\n", " yield circ"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def run_tomography_circuits(state, qubits, bits, backend):\n", " circs = list(gen_tomography_circuits(state, qubits, bits))\n", " # Compile and run each circuit\n", " circs = backend.get_compiled_circuits(circs)\n", " handles = backend.process_circuits(circs, n_shots=2000)\n", " # Get the observed measurement probabilities\n", " probs_list = []\n", " for result in backend.get_results(handles):\n", " counts = result.get_counts()\n", " probs = probs_from_counts(counts)\n", " probs_list.append(probs)\n", " return probs_list"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def fit_tomography_outcomes(probs_list, n_qbs):\n", " # Define the density matrices for the basis states\n", " basis = dict()\n", " basis[(Pauli.X, 0)] = np.asarray([[0.5, 0.5], [0.5, 0.5]])\n", " basis[(Pauli.X, 1)] = np.asarray([[0.5, -0.5], [-0.5, 0.5]])\n", " basis[(Pauli.Y, 0)] = np.asarray([[0.5, -0.5j], [0.5j, 0.5]])\n", " basis[(Pauli.Y, 1)] = np.asarray([[0.5, 0.5j], [-0.5j, 0.5]])\n", " basis[(Pauli.Z, 0)] = np.asarray([[1, 0], [0, 0]])\n", " basis[(Pauli.Z, 1)] = np.asarray([[0, 0], [0, 1]])\n", " dim = 2**n_qbs\n", " # Define vector all_probs as a concatenation of probability vectors for each measurement (2**n x 3**n, 1)\n", " # Define matrix all_ops mapping a (vectorised) density matrix to a vector of probabilities for each measurement\n", " # (2**n x 3**n, 2**n x 2**n)\n", " all_probs = []\n", " all_ops = []\n", " for paulis, probs in zip(\n", " product([Pauli.X, Pauli.Y, Pauli.Z], repeat=n_qbs), probs_list\n", " ):\n", " prob_vec = []\n", " meas_ops = []\n", " for outcome in product([0, 1], repeat=n_qbs):\n", " prob_vec.append(probs.get(outcome, 0))\n", " op = np.eye(1, dtype=complex)\n", " for p, o in zip(paulis, outcome):\n", " op = np.kron(op, basis[(p, o)])\n", " meas_ops.append(op.reshape(1, dim * dim).conj())\n", " all_probs.append(np.vstack(prob_vec))\n", " all_ops.append(np.vstack(meas_ops))\n", " # Solve for density matrix by minimising || all_ops * dm - all_probs ||\n", " dm, _, _, _ = lstsq(np.vstack(all_ops), np.vstack(all_probs))\n", " dm = dm.reshape(dim, dim)\n", " # Make density matrix positive semi-definite\n", " v, w = eigh(dm)\n", " for i in range(dim):\n", " if v[i] < 0:\n", " for j in range(i + 1, dim):\n", " v[j] += v[i] / (dim - (i + 1))\n", " v[i] = 0\n", " dm = np.zeros([dim, dim], dtype=complex)\n", " for j in range(dim):\n", " dm += v[j] * np.outer(w[:, j], np.conj(w[:, j]))\n", " # Normalise trace of density matrix\n", " dm /= np.trace(dm)\n", " return dm"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["probs_list = run_tomography_circuits(\n", " es, [ava[0], charlie[0]], [data[0], data[1]], backend\n", ")\n", "dm = fit_tomography_outcomes(probs_list, 2)\n", "print(dm.round(3))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is very close to the true density matrix for a pure Bell state. We can attribute the error here to the sampling error since we only take 2000 samples of each measurement circuit.
\n", "
\n", "To quantify exactly how similar it is to the correct density matrix, we can calculate the fidelity."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from scipy.linalg import sqrtm"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def fidelity(dm0, dm1):\n", " # Calculate the fidelity between two density matrices\n", " sq0 = sqrtm(dm0)\n", " sq1 = sqrtm(dm1)\n", " return np.linalg.norm(sq0.dot(sq1)) ** 2"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["bell_state = np.asarray(\n", " [\n", " [0.5, 0, 0, 0.5],\n", " [0, 0, 0, 0],\n", " [0, 0, 0, 0],\n", " [0.5, 0, 0, 0.5],\n", " ]\n", ")\n", "print(fidelity(dm, bell_state))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This high fidelity is unsurprising since we have a completely noiseless simulation. So the next step is to add some noise to the simulation and observe how the overall fidelity is affected. The `AerBackend` wraps around the Qiskit Aer simulator and can pass on any `qiskit.providers.aer.noise.NoiseModel` to the simulator. Let's start by adding some uniform depolarising noise to each CX gate and some uniform measurement error."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from qiskit.providers.aer.noise import NoiseModel, depolarizing_error, ReadoutError"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def make_noise_model(dep_err_rate, ro_err_rate, qubits):\n", " # Define a noise model that applies uniformly to the given qubits\n", " model = NoiseModel()\n", " dep_err = depolarizing_error(dep_err_rate, 2)\n", " ro_err = ReadoutError(\n", " [[1 - ro_err_rate, ro_err_rate], [ro_err_rate, 1 - ro_err_rate]]\n", " )\n", " # Add depolarising error to CX gates between any qubits (implying full connectivity)\n", " for i, j in product(qubits, repeat=2):\n", " if i != j:\n", " model.add_quantum_error(dep_err, [\"cx\"], [i, j])\n", " # Add readout error for each qubit\n", " for i in qubits:\n", " model.add_readout_error(ro_err, qubits=[i])\n", " return model"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_model = make_noise_model(0.03, 0.05, range(4))\n", "backend = AerBackend(noise_model=test_model)\n", "probs_list = run_tomography_circuits(\n", " es, [ava[0], charlie[0]], [data[0], data[1]], backend\n", ")\n", "dm = fit_tomography_outcomes(probs_list, 2)\n", "print(dm.round(3))\n", "print(fidelity(dm, bell_state))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Despite the very small circuit and the relatively small error rates, the fidelity of the final state has reduced considerably.
\n", "
\n", "As far as circuits go, the entanglement swapping protocol is little more than a toy example and is nothing close to the scale of circuits for most interesting quantum computational problems. However, it is possible to iterate the protocol many times to build up a larger computation, allowing us to see the impact of the noise at different scales."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import OpType\n", "from plotly.graph_objects import Scatter, Figure"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def iterated_entanglement_swap(n_iter):\n", " # Iterate the entanglement swapping protocol n_iter times\n", " it_es = Circuit()\n", " ava = it_es.add_q_register(\"a\", 1)\n", " bella = it_es.add_q_register(\"b\", 2)\n", " charlie = it_es.add_q_register(\"c\", 1)\n", " data = it_es.add_c_register(\"d\", 2)\n\n", " # Start with an initial Bell state\n", " it_es.H(ava[0])\n", " it_es.CX(ava[0], bella[0])\n", " for i in range(n_iter):\n", " if i % 2 == 0:\n", " # Teleport bella[0] to charlie[0] to give a Bell pair between ava[0] and charlier[0]\n", " tel_to_c = qtel.copy()\n", " tel_to_c.rename_units(\n", " {alice[0]: bella[0], alice[1]: bella[1], bob[0]: charlie[0]}\n", " )\n", " it_es.append(tel_to_c)\n", " it_es.add_gate(OpType.Reset, [bella[0]])\n", " it_es.add_gate(OpType.Reset, [bella[1]])\n", " else:\n", " # Teleport charlie[0] to bella[0] to give a Bell pair between ava[0] and bella[0]\n", " tel_to_b = qtel.copy()\n", " tel_to_b.rename_units(\n", " {alice[0]: charlie[0], alice[1]: bella[1], bob[0]: bella[0]}\n", " )\n", " it_es.append(tel_to_b)\n", " it_es.add_gate(OpType.Reset, [bella[1]])\n", " it_es.add_gate(OpType.Reset, [charlie[0]])\n", " # Return the circuit and the qubits expected to share a Bell pair\n", " if n_iter % 2 == 0:\n", " return it_es, [ava[0], bella[0]]\n", " else:\n", " return it_es, [ava[0], charlie[0]]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def iterated_noisy_experiment(dep_err_rate, ro_err_rate, max_iter):\n", " # Set up the noisy simulator with the given error rates\n", " test_model = make_noise_model(dep_err_rate, ro_err_rate, range(4))\n", " backend = AerBackend(noise_model=test_model)\n", " # Estimate the fidelity after n iterations, from 0 to max_iter (inclusive)\n", " fid_list = []\n", " for i in range(max_iter + 1):\n", " it_es, qubits = iterated_entanglement_swap(i)\n", " probs_list = run_tomography_circuits(it_es, qubits, [data[0], data[1]], backend)\n", " dm = fit_tomography_outcomes(probs_list, 2)\n", " fid = fidelity(dm, bell_state)\n", " fid_list.append(fid)\n", " return fid_list"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["fig = Figure()\n", "fig.update_layout(\n", " title=\"Iterated Entanglement Swapping under Noise (dep_err = 0.03)\",\n", " xaxis_title=\"Iterations\",\n", " xaxis=dict(range=[0, 10]),\n", " yaxis_title=\"Fidelity\",\n", ")\n", "iter_range = np.arange(11)\n", "for i in range(7):\n", " fids = iterated_noisy_experiment(0.03, 0.025 * i, 10)\n", " plot_data = Scatter(\n", " x=iter_range, y=fids, name=\"ro_err=\" + str(np.round(0.025 * i, 3))\n", " )\n", " fig.add_trace(plot_data)\n", "try:\n", " fig.show(renderer=\"svg\")\n", "except ValueError as e:\n", " print(e) # requires plotly-orca"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["fig = Figure()\n", "fig.update_layout(\n", " title=\"Iterated Entanglement Swapping under Noise (ro_err = 0.05)\",\n", " xaxis_title=\"Iterations\",\n", " xaxis=dict(range=[0, 10]),\n", " yaxis_title=\"Fidelity\",\n", ")\n", "iter_range = np.arange(11)\n", "for i in range(9):\n", " fids = iterated_noisy_experiment(0.01 * i, 0.05, 10)\n", " plot_data = Scatter(\n", " x=iter_range, y=fids, name=\"dep_err=\" + str(np.round(0.01 * i, 3))\n", " )\n", " fig.add_trace(plot_data)\n", "try:\n", " fig.show(renderer=\"svg\")\n", "except ValueError as e:\n", " print(e) # requires plotly-orca"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These graphs are not very surprising, but are still important for seeing that the current error rates of typical NISQ devices become crippling for fidelities very quickly after repeated mid-circuit measurements and corrections (even with this overly-simplified model with uniform noise and no crosstalk or higher error modes). This provides good motivation for the adoption of error mitigation techniques, and for the development of new techniques that are robust to errors in mid-circuit measurements."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Exercises:
\n", "- Vary the fixed noise levels to compare how impactful the depolarising and measurement errors are.
\n", "- Add extra noise characteristics to the noise model to obtain something that more resembles a real device. Possible options include adding error during the reset operations, extending the errors to be non-local, or constructing the noise model from a device's calibration data.
\n", "- Change the circuit from iterated entanglement swapping to iterated applications of a correction circuit from a simple error-correcting code. Do you expect this to be more sensitive to depolarising errors from unitary gates or measurement errors?"]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file diff --git a/examples/ucc_vqe.ipynb b/examples/ucc_vqe.ipynb index 99250907..5400c6c3 100644 --- a/examples/ucc_vqe.ipynb +++ b/examples/ucc_vqe.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# VQE for Unitary Coupled Cluster using tket"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In this tutorial, we will focus on:
\n", "- building parameterised ans\u00e4tze for variational algorithms;
\n", "- compilation tools for UCC-style ans\u00e4tze."]}, {"cell_type": "markdown", "metadata": {}, "source": ["This example assumes the reader is familiar with the Variational Quantum Eigensolver and its application to electronic structure problems through the Unitary Coupled Cluster approach.
\n", "
\n", "To run this example, you will need `pytket` and `pytket-qiskit`, as well as `openfermion`, `scipy`, and `sympy`.
\n", "
\n", "We will start with a basic implementation and then gradually modify it to make it faster, more general, and less noisy. The final solution is given in full at the bottom of the notebook.
\n", "
\n", "Suppose we have some electronic configuration problem, expressed via a physical Hamiltonian. (The Hamiltonian and excitations in this example were obtained using `qiskit-aqua` version 0.5.2 and `pyscf` for H2, bond length 0.75A, sto3g basis, Jordan-Wigner encoding, with no qubit reduction or orbital freezing.). We express it succinctly using the openfermion library:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import openfermion as of"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian = (\n", " -0.8153001706270075 * of.QubitOperator(\"\")\n", " + 0.16988452027940318 * of.QubitOperator(\"Z0\")\n", " + -0.21886306781219608 * of.QubitOperator(\"Z1\")\n", " + 0.16988452027940323 * of.QubitOperator(\"Z2\")\n", " + -0.2188630678121961 * of.QubitOperator(\"Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z0 Z1\")\n", " + 0.16821198673715723 * of.QubitOperator(\"Z0 Z2\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z0 Z3\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z1 Z2\")\n", " + 0.1739537877649417 * of.QubitOperator(\"Z1 Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z2 Z3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 Y2 Y3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 Y2 Y3\")\n", ")\n", "nuclear_repulsion_energy = 0.70556961456"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We would like to define our ansatz for arbitrary parameter values. For simplicity, let's start with a Hardware Efficient Ansatz."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Hardware efficient ansatz:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def hea(params):\n", " ansatz = Circuit(4)\n", " for i in range(4):\n", " ansatz.Ry(params[i], i)\n", " for i in range(3):\n", " ansatz.CX(i, i + 1)\n", " for i in range(4):\n", " ansatz.Ry(params[4 + i], i)\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can use this to build the objective function for our optimisation."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import AerBackend\n", "from pytket.utils.expectations import expectation_from_counts"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Naive objective function:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " energy = 0\n", " for term, coeff in hamiltonian.terms.items():\n", " if not term:\n", " energy += coeff\n", " continue\n", " circ = hea(params)\n", " circ.add_c_register(\"c\", len(term))\n", " for i, (q, pauli) in enumerate(term):\n", " if pauli == \"X\":\n", " circ.H(q)\n", " elif pauli == \"Y\":\n", " circ.V(q)\n", " circ.Measure(q, i)\n", " compiled_circ = backend.get_compiled_circuit(circ)\n", " counts = backend.run_circuit(compiled_circ, n_shots=4000).get_counts()\n", " energy += coeff * expectation_from_counts(counts)\n", " return energy + nuclear_repulsion_energy"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This objective function is then run through a classical optimiser to find the set of parameter values that minimise the energy of the system. For the sake of example, we will just run this with a single parameter value."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["arg_values = [\n", " -7.31158201e-02,\n", " -1.64514836e-04,\n", " 1.12585591e-03,\n", " -2.58367544e-03,\n", " 1.00006068e00,\n", " -1.19551357e-03,\n", " 9.99963988e-01,\n", " 2.53283285e-03,\n", "]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["energy = objective(arg_values)\n", "print(energy)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The HEA is designed to cram as many orthogonal degrees of freedom into a small circuit as possible to be able to explore a large region of the Hilbert space whilst the circuits themselves can be run with minimal noise. These ans\u00e4tze give virtually-optimal circuits by design, but suffer from an excessive number of variational parameters making convergence slow, barren plateaus where the classical optimiser fails to make progress, and spanning a space where most states lack a physical interpretation. These drawbacks can necessitate adding penalties and may mean that the ansatz cannot actually express the true ground state.
\n", "
\n", "The UCC ansatz, on the other hand, is derived from the electronic configuration. It sacrifices efficiency of the circuit for the guarantee of physical states and the variational parameters all having some meaningful effect, which helps the classical optimisation to converge.
\n", "
\n", "This starts by defining the terms of our single and double excitations. These would usually be generated using the orbital configurations, so we will just use a hard-coded example here for the purposes of demonstration."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import Qubit\n", "from pytket.pauli import Pauli, QubitPauliString"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["q = [Qubit(i) for i in range(4)]\n", "xyii = QubitPauliString([q[0], q[1]], [Pauli.X, Pauli.Y])\n", "yxii = QubitPauliString([q[0], q[1]], [Pauli.Y, Pauli.X])\n", "iixy = QubitPauliString([q[2], q[3]], [Pauli.X, Pauli.Y])\n", "iiyx = QubitPauliString([q[2], q[3]], [Pauli.Y, Pauli.X])\n", "xxxy = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.X, Pauli.Y])\n", "xxyx = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.Y, Pauli.X])\n", "xyxx = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.X, Pauli.X])\n", "yxxx = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.X, Pauli.X])\n", "yyyx = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.Y, Pauli.X])\n", "yyxy = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.X, Pauli.Y])\n", "yxyy = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.Y, Pauli.Y])\n", "xyyy = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.Y, Pauli.Y])"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["singles_a = {xyii: 1.0, yxii: -1.0}\n", "singles_b = {iixy: 1.0, iiyx: -1.0}\n", "doubles = {\n", " xxxy: 0.25,\n", " xxyx: -0.25,\n", " xyxx: 0.25,\n", " yxxx: -0.25,\n", " yyyx: -0.25,\n", " yyxy: 0.25,\n", " yxyy: -0.25,\n", " xyyy: 0.25,\n", "}"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Building the ansatz circuit itself is often done naively by defining the map from each term down to basic gates and then applying it to each term."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def add_operator_term(circuit: Circuit, term: QubitPauliString, angle: float):\n", " qubits = []\n", " for q, p in term.map.items():\n", " if p != Pauli.I:\n", " qubits.append(q)\n", " if p == Pauli.X:\n", " circuit.H(q)\n", " elif p == Pauli.Y:\n", " circuit.V(q)\n", " for i in range(len(qubits) - 1):\n", " circuit.CX(i, i + 1)\n", " circuit.Rz(angle, len(qubits) - 1)\n", " for i in reversed(range(len(qubits) - 1)):\n", " circuit.CX(i, i + 1)\n", " for q, p in term.map.items():\n", " if p == Pauli.X:\n", " circuit.H(q)\n", " elif p == Pauli.Y:\n", " circuit.Vdg(q)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Unitary Coupled Cluster Singles & Doubles ansatz:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def ucc(params):\n", " ansatz = Circuit(4)\n", " # Set initial reference state\n", " ansatz.X(1).X(3)\n", " # Evolve by excitations\n", " for term, coeff in singles_a.items():\n", " add_operator_term(ansatz, term, coeff * params[0])\n", " for term, coeff in singles_b.items():\n", " add_operator_term(ansatz, term, coeff * params[1])\n", " for term, coeff in doubles.items():\n", " add_operator_term(ansatz, term, coeff * params[2])\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is already quite verbose, but `pytket` has a neat shorthand construction for these operator terms using the `PauliExpBox` construction. We can then decompose these into basic gates using the `DecomposeBoxes` compiler pass."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import PauliExpBox\n", "from pytket.passes import DecomposeBoxes"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def add_excitation(circ, term_dict, param):\n", " for term, coeff in term_dict.items():\n", " qubits, paulis = zip(*term.map.items())\n", " pbox = PauliExpBox(paulis, coeff * param)\n", " circ.add_pauliexpbox(pbox, qubits)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["UCC ansatz with syntactic shortcuts:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def ucc(params):\n", " ansatz = Circuit(4)\n", " ansatz.X(1).X(3)\n", " add_excitation(ansatz, singles_a, params[0])\n", " add_excitation(ansatz, singles_b, params[1])\n", " add_excitation(ansatz, doubles, params[2])\n", " DecomposeBoxes().apply(ansatz)\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The objective function can also be simplified using a utility method for constructing the measurement circuits and processing for expectation value calculations. For that, we convert the Hamiltonian to a pytket QubitPauliOperator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["pauli_sym = {\"I\": Pauli.I, \"X\": Pauli.X, \"Y\": Pauli.Y, \"Z\": Pauli.Z}"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qps_from_openfermion(paulis):\n", " \"\"\"Convert OpenFermion tensor of Paulis to pytket QubitPauliString.\"\"\"\n", " qlist = []\n", " plist = []\n", " for q, p in paulis:\n", " qlist.append(Qubit(q))\n", " plist.append(pauli_sym[p])\n", " return QubitPauliString(qlist, plist)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qpo_from_openfermion(openf_op):\n", " \"\"\"Convert OpenFermion QubitOperator to pytket QubitPauliOperator.\"\"\"\n", " tk_op = dict()\n", " for term, coeff in openf_op.terms.items():\n", " string = qps_from_openfermion(term)\n", " tk_op[string] = coeff\n", " return QubitPauliOperator(tk_op)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian_op = qpo_from_openfermion(hamiltonian)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Simplified objective function using utilities:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils.expectations import get_operator_expectation_value"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc(params)\n", " return (\n", " get_operator_expectation_value(circ, hamiltonian_op, backend, n_shots=4000)\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["arg_values = [-3.79002933e-05, 2.42964799e-05, 4.63447157e-01]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["energy = objective(arg_values)\n", "print(energy)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is now the simplest form that this operation can take, but it isn't necessarily the most effective. When we decompose the ansatz circuit into basic gates, it is still very expensive. We can employ some of the circuit simplification passes available in `pytket` to reduce its size and improve fidelity in practice.
\n", "
\n", "A good example is to decompose each `PauliExpBox` into basic gates and then apply `FullPeepholeOptimise`, which defines a compilation strategy utilising all of the simplifications in `pytket` that act locally on small regions of a circuit. We can examine the effectiveness by looking at the number of two-qubit gates before and after simplification, which tends to be a good indicator of fidelity for near-term systems where these gates are often slow and inaccurate."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import OpType\n", "from pytket.passes import FullPeepholeOptimise"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_circuit = ucc(arg_values)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count before\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth before\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["FullPeepholeOptimise().apply(test_circuit)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count after FPO\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth after FPO\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These simplification techniques are very general and are almost always beneficial to apply to a circuit if you want to eliminate local redundancies. But UCC ans\u00e4tze have extra structure that we can exploit further. They are defined entirely out of exponentiated tensors of Pauli matrices, giving the regular structure described by the `PauliExpBox`es. Under many circumstances, it is more efficient to not synthesise these constructions individually, but simultaneously in groups. The `PauliSimp` pass finds the description of a given circuit as a sequence of `PauliExpBox`es and resynthesises them (by default, in groups of commuting terms). This can cause great change in the overall structure and shape of the circuit, enabling the identification and elimination of non-local redundancy."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.passes import PauliSimp"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_circuit = ucc(arg_values)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count before\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth before\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["PauliSimp().apply(test_circuit)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count after PS\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth after PS\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["FullPeepholeOptimise().apply(test_circuit)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count after PS+FPO\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth after PS+FPO\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["To include this into our routines, we can just add the simplification passes to the objective function. The `get_operator_expectation_value` utility handles compiling to meet the requirements of the backend, so we don't have to worry about that here."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function with circuit simplification:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc(params)\n", " PauliSimp().apply(circ)\n", " FullPeepholeOptimise().apply(circ)\n", " return (\n", " get_operator_expectation_value(circ, hamiltonian_op, backend, n_shots=4000)\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These circuit simplification techniques have tried to preserve the exact unitary of the circuit, but there are ways to change the unitary whilst preserving the correctness of the algorithm as a whole.
\n", "
\n", "For example, the excitation terms are generated by trotterisation of the excitation operator, and the order of the terms does not change the unitary in the limit of many trotter steps, so in this sense we are free to sequence the terms how we like and it is sensible to do this in a way that enables efficient synthesis of the circuit. Prioritising collecting terms into commuting sets is a very beneficial heuristic for this and can be performed using the `gen_term_sequence_circuit` method to group the terms together into collections of `PauliExpBox`es and the `GuidedPauliSimp` pass to utilise these sets for synthesis."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.passes import GuidedPauliSimp\n", "from pytket.utils import gen_term_sequence_circuit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def ucc(params):\n", " singles_params = {qps: params[0] * coeff for qps, coeff in singles.items()}\n", " doubles_params = {qps: params[1] * coeff for qps, coeff in doubles.items()}\n", " excitation_op = QubitPauliOperator({**singles_params, **doubles_params})\n", " reference_circ = Circuit(4).X(1).X(3)\n", " ansatz = gen_term_sequence_circuit(excitation_op, reference_circ)\n", " GuidedPauliSimp().apply(ansatz)\n", " FullPeepholeOptimise().apply(ansatz)\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Adding these simplification routines doesn't come for free. Compiling and simplifying the circuit to achieve the best results possible can be a difficult task, which can take some time for the classical computer to perform.
\n", "
\n", "During a VQE run, we will call this objective function many times and run many measurement circuits within each, but the circuits that are run on the quantum computer are almost identical, having the same gate structure but with different gate parameters and measurements. We have already exploited this within the body of the objective function by simplifying the ansatz circuit before we call `get_operator_expectation_value`, so it is only done once per objective calculation rather than once per measurement circuit.
\n", "
\n", "We can go even further by simplifying it once outside of the objective function, and then instantiating the simplified ansatz with the parameter values needed. For this, we will construct the UCC ansatz circuit using symbolic (parametric) gates."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from sympy import symbols"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Symbolic UCC ansatz generation:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["syms = symbols(\"p0 p1 p2\")\n", "singles_a_syms = {qps: syms[0] * coeff for qps, coeff in singles_a.items()}\n", "singles_b_syms = {qps: syms[1] * coeff for qps, coeff in singles_b.items()}\n", "doubles_syms = {qps: syms[2] * coeff for qps, coeff in doubles.items()}\n", "excitation_op = QubitPauliOperator({**singles_a_syms, **singles_b_syms, **doubles_syms})\n", "ucc_ref = Circuit(4).X(1).X(3)\n", "ucc = gen_term_sequence_circuit(excitation_op, ucc_ref)\n", "GuidedPauliSimp().apply(ucc)\n", "FullPeepholeOptimise().apply(ucc)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function using the symbolic ansatz:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc.copy()\n", " sym_map = dict(zip(syms, params))\n", " circ.symbol_substitution(sym_map)\n", " return (\n", " get_operator_expectation_value(circ, hamiltonian_op, backend, n_shots=4000)\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We have now got some very good use of `pytket` for simplifying each individual circuit used in our experiment and for minimising the amount of time spent compiling, but there is still more we can do in terms of reducing the amount of work the quantum computer has to do. Currently, each (non-trivial) term in our measurement hamiltonian is measured by a different circuit within each expectation value calculation. Measurement reduction techniques exist for identifying when these observables commute and hence can be simultaneously measured, reducing the number of circuits required for the full expectation value calculation.
\n", "
\n", "This is built in to the `get_operator_expectation_value` method and can be applied by specifying a way to partition the measuremrnt terms. `PauliPartitionStrat.CommutingSets` can greatly reduce the number of measurement circuits by combining any number of terms that mutually commute. However, this involves potentially adding an arbitrary Clifford circuit to change the basis of the measurements which can be costly on NISQ devices, so `PauliPartitionStrat.NonConflictingSets` trades off some of the reduction in circuit number to guarantee that only single-qubit gates are introduced."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.partition import PauliPartitionStrat"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function using measurement reduction:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc.copy()\n", " sym_map = dict(zip(syms, params))\n", " circ.symbol_substitution(sym_map)\n", " return (\n", " get_operator_expectation_value(\n", " circ,\n", " operator,\n", " backend,\n", " n_shots=4000,\n", " partition_strat=PauliPartitionStrat.CommutingSets,\n", " )\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "markdown", "metadata": {}, "source": ["At this point, we have completely transformed how our VQE objective function works, improving its resilience to noise, cutting the number of circuits run, and maintaining fast runtimes. In doing this, we have explored a number of the features `pytket` offers that are beneficial to VQE and the UCC method:
\n", "- high-level syntactic constructs for evolution operators;
\n", "- utility methods for easy expectation value calculations;
\n", "- both generic and domain-specific circuit simplification methods;
\n", "- symbolic circuit compilation;
\n", "- measurement reduction for expectation value calculations."]}, {"cell_type": "markdown", "metadata": {}, "source": ["For the sake of completeness, the following gives the full code for the final solution, including passing the objective function to a classical optimiser to find the ground state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import openfermion as of\n", "from scipy.optimize import minimize\n", "from sympy import symbols"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import AerBackend\n", "from pytket.circuit import Circuit, Qubit\n", "from pytket.partition import PauliPartitionStrat\n", "from pytket.passes import GuidedPauliSimp, FullPeepholeOptimise\n", "from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.utils import get_operator_expectation_value, gen_term_sequence_circuit\n", "from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Obtain electronic Hamiltonian:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian = (\n", " -0.8153001706270075 * of.QubitOperator(\"\")\n", " + 0.16988452027940318 * of.QubitOperator(\"Z0\")\n", " + -0.21886306781219608 * of.QubitOperator(\"Z1\")\n", " + 0.16988452027940323 * of.QubitOperator(\"Z2\")\n", " + -0.2188630678121961 * of.QubitOperator(\"Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z0 Z1\")\n", " + 0.16821198673715723 * of.QubitOperator(\"Z0 Z2\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z0 Z3\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z1 Z2\")\n", " + 0.1739537877649417 * of.QubitOperator(\"Z1 Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z2 Z3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 Y2 Y3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 Y2 Y3\")\n", ")\n", "nuclear_repulsion_energy = 0.70556961456"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["pauli_sym = {\"I\": Pauli.I, \"X\": Pauli.X, \"Y\": Pauli.Y, \"Z\": Pauli.Z}"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qps_from_openfermion(paulis):\n", " \"\"\"Convert OpenFermion tensor of Paulis to pytket QubitPauliString.\"\"\"\n", " qlist = []\n", " plist = []\n", " for q, p in paulis:\n", " qlist.append(Qubit(q))\n", " plist.append(pauli_sym[p])\n", " return QubitPauliString(qlist, plist)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qpo_from_openfermion(openf_op):\n", " \"\"\"Convert OpenFermion QubitOperator to pytket QubitPauliOperator.\"\"\"\n", " tk_op = dict()\n", " for term, coeff in openf_op.terms.items():\n", " string = qps_from_openfermion(term)\n", " tk_op[string] = coeff\n", " return QubitPauliOperator(tk_op)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian_op = qpo_from_openfermion(hamiltonian)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Obtain terms for single and double excitations:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["q = [Qubit(i) for i in range(4)]\n", "xyii = QubitPauliString([q[0], q[1]], [Pauli.X, Pauli.Y])\n", "yxii = QubitPauliString([q[0], q[1]], [Pauli.Y, Pauli.X])\n", "iixy = QubitPauliString([q[2], q[3]], [Pauli.X, Pauli.Y])\n", "iiyx = QubitPauliString([q[2], q[3]], [Pauli.Y, Pauli.X])\n", "xxxy = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.X, Pauli.Y])\n", "xxyx = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.Y, Pauli.X])\n", "xyxx = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.X, Pauli.X])\n", "yxxx = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.X, Pauli.X])\n", "yyyx = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.Y, Pauli.X])\n", "yyxy = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.X, Pauli.Y])\n", "yxyy = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.Y, Pauli.Y])\n", "xyyy = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.Y, Pauli.Y])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Symbolic UCC ansatz generation:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["syms = symbols(\"p0 p1 p2\")\n", "singles_syms = {xyii: syms[0], yxii: -syms[0], iixy: syms[1], iiyx: -syms[1]}\n", "doubles_syms = {\n", " xxxy: 0.25 * syms[2],\n", " xxyx: -0.25 * syms[2],\n", " xyxx: 0.25 * syms[2],\n", " yxxx: -0.25 * syms[2],\n", " yyyx: -0.25 * syms[2],\n", " yyxy: 0.25 * syms[2],\n", " yxyy: -0.25 * syms[2],\n", " xyyy: 0.25 * syms[2],\n", "}\n", "excitation_op = QubitPauliOperator({**singles_syms, **doubles_syms})\n", "ucc_ref = Circuit(4).X(0).X(2)\n", "ucc = gen_term_sequence_circuit(excitation_op, ucc_ref)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Circuit simplification:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["GuidedPauliSimp().apply(ucc)\n", "FullPeepholeOptimise().apply(ucc)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Connect to a simulator/device:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc.copy()\n", " sym_map = dict(zip(syms, params))\n", " circ.symbol_substitution(sym_map)\n", " return (\n", " get_operator_expectation_value(\n", " circ,\n", " hamiltonian_op,\n", " backend,\n", " n_shots=4000,\n", " partition_strat=PauliPartitionStrat.CommutingSets,\n", " )\n", " + nuclear_repulsion_energy\n", " ).real"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Optimise against the objective function:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["initial_params = [1e-4, 1e-4, 4e-1]\n", "# #result = minimize(objective, initial_params, method=\"Nelder-Mead\")\n", "# #print(\"Final parameter values\", result.x)\n", "# #print(\"Final energy value\", result.fun)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Exercises:
\n", "- Replace the `get_operator_expectation_value` call with its implementation and use this to pull the analysis for measurement reduction outside of the objective function, so our circuits can be fully determined and compiled once. This means that the `symbol_substitution` method will need to be applied to each measurement circuit instead of just the state preparation circuit.
\n", "- Use the `SpamCorrecter` class to add some mitigation of the measurement errors. Start by running the characterisation circuits first, before your main VQE loop, then apply the mitigation to each of the circuits run within the objective function.
\n", "- Change the `backend` by passing in a `Qiskit` `NoiseModel` to simulate a noisy device. Compare the accuracy of the objective function both with and without the circuit simplification. Try running a classical optimiser over the objective function and compare the convergence rates with different noise models. If you have access to a QPU, try changing the `backend` to connect to that and compare the results to the simulator."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# VQE for Unitary Coupled Cluster using TKET"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In this tutorial, we will focus on:
\n", "- building parameterised ans\u00e4tze for variational algorithms;
\n", "- compilation tools for UCC-style ans\u00e4tze."]}, {"cell_type": "markdown", "metadata": {}, "source": ["This example assumes the reader is familiar with the Variational Quantum Eigensolver and its application to electronic structure problems through the Unitary Coupled Cluster approach.
\n", "
\n", "To run this example, you will need `pytket` and `pytket-qiskit`, as well as `openfermion`, `scipy`, and `sympy`.
\n", "
\n", "We will start with a basic implementation and then gradually modify it to make it faster, more general, and less noisy. The final solution is given in full at the bottom of the notebook.
\n", "
\n", "Suppose we have some electronic configuration problem, expressed via a physical Hamiltonian. (The Hamiltonian and excitations in this example were obtained using `qiskit-aqua` version 0.5.2 and `pyscf` for H2, bond length 0.75A, sto3g basis, Jordan-Wigner encoding, with no qubit reduction or orbital freezing.). We express it succinctly using the openfermion library:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import openfermion as of"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian = (\n", " -0.8153001706270075 * of.QubitOperator(\"\")\n", " + 0.16988452027940318 * of.QubitOperator(\"Z0\")\n", " + -0.21886306781219608 * of.QubitOperator(\"Z1\")\n", " + 0.16988452027940323 * of.QubitOperator(\"Z2\")\n", " + -0.2188630678121961 * of.QubitOperator(\"Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z0 Z1\")\n", " + 0.16821198673715723 * of.QubitOperator(\"Z0 Z2\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z0 Z3\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z1 Z2\")\n", " + 0.1739537877649417 * of.QubitOperator(\"Z1 Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z2 Z3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 Y2 Y3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 Y2 Y3\")\n", ")\n", "nuclear_repulsion_energy = 0.70556961456"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We would like to define our ansatz for arbitrary parameter values. For simplicity, let's start with a Hardware Efficient Ansatz."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Hardware efficient ansatz:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def hea(params):\n", " ansatz = Circuit(4)\n", " for i in range(4):\n", " ansatz.Ry(params[i], i)\n", " for i in range(3):\n", " ansatz.CX(i, i + 1)\n", " for i in range(4):\n", " ansatz.Ry(params[4 + i], i)\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can use this to build the objective function for our optimisation."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import AerBackend\n", "from pytket.utils.expectations import expectation_from_counts"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Naive objective function:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " energy = 0\n", " for term, coeff in hamiltonian.terms.items():\n", " if not term:\n", " energy += coeff\n", " continue\n", " circ = hea(params)\n", " circ.add_c_register(\"c\", len(term))\n", " for i, (q, pauli) in enumerate(term):\n", " if pauli == \"X\":\n", " circ.H(q)\n", " elif pauli == \"Y\":\n", " circ.V(q)\n", " circ.Measure(q, i)\n", " compiled_circ = backend.get_compiled_circuit(circ)\n", " counts = backend.run_circuit(compiled_circ, n_shots=4000).get_counts()\n", " energy += coeff * expectation_from_counts(counts)\n", " return energy + nuclear_repulsion_energy"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This objective function is then run through a classical optimiser to find the set of parameter values that minimise the energy of the system. For the sake of example, we will just run this with a single parameter value."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["arg_values = [\n", " -7.31158201e-02,\n", " -1.64514836e-04,\n", " 1.12585591e-03,\n", " -2.58367544e-03,\n", " 1.00006068e00,\n", " -1.19551357e-03,\n", " 9.99963988e-01,\n", " 2.53283285e-03,\n", "]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["energy = objective(arg_values)\n", "print(energy)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The HEA is designed to cram as many orthogonal degrees of freedom into a small circuit as possible to be able to explore a large region of the Hilbert space whilst the circuits themselves can be run with minimal noise. These ans\u00e4tze give virtually-optimal circuits by design, but suffer from an excessive number of variational parameters making convergence slow, barren plateaus where the classical optimiser fails to make progress, and spanning a space where most states lack a physical interpretation. These drawbacks can necessitate adding penalties and may mean that the ansatz cannot actually express the true ground state.
\n", "
\n", "The UCC ansatz, on the other hand, is derived from the electronic configuration. It sacrifices efficiency of the circuit for the guarantee of physical states and the variational parameters all having some meaningful effect, which helps the classical optimisation to converge.
\n", "
\n", "This starts by defining the terms of our single and double excitations. These would usually be generated using the orbital configurations, so we will just use a hard-coded example here for the purposes of demonstration."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import Qubit\n", "from pytket.pauli import Pauli, QubitPauliString"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["q = [Qubit(i) for i in range(4)]\n", "xyii = QubitPauliString([q[0], q[1]], [Pauli.X, Pauli.Y])\n", "yxii = QubitPauliString([q[0], q[1]], [Pauli.Y, Pauli.X])\n", "iixy = QubitPauliString([q[2], q[3]], [Pauli.X, Pauli.Y])\n", "iiyx = QubitPauliString([q[2], q[3]], [Pauli.Y, Pauli.X])\n", "xxxy = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.X, Pauli.Y])\n", "xxyx = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.Y, Pauli.X])\n", "xyxx = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.X, Pauli.X])\n", "yxxx = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.X, Pauli.X])\n", "yyyx = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.Y, Pauli.X])\n", "yyxy = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.X, Pauli.Y])\n", "yxyy = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.Y, Pauli.Y])\n", "xyyy = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.Y, Pauli.Y])"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["singles_a = {xyii: 1.0, yxii: -1.0}\n", "singles_b = {iixy: 1.0, iiyx: -1.0}\n", "doubles = {\n", " xxxy: 0.25,\n", " xxyx: -0.25,\n", " xyxx: 0.25,\n", " yxxx: -0.25,\n", " yyyx: -0.25,\n", " yyxy: 0.25,\n", " yxyy: -0.25,\n", " xyyy: 0.25,\n", "}"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Building the ansatz circuit itself is often done naively by defining the map from each term down to basic gates and then applying it to each term."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def add_operator_term(circuit: Circuit, term: QubitPauliString, angle: float):\n", " qubits = []\n", " for q, p in term.map.items():\n", " if p != Pauli.I:\n", " qubits.append(q)\n", " if p == Pauli.X:\n", " circuit.H(q)\n", " elif p == Pauli.Y:\n", " circuit.V(q)\n", " for i in range(len(qubits) - 1):\n", " circuit.CX(i, i + 1)\n", " circuit.Rz(angle, len(qubits) - 1)\n", " for i in reversed(range(len(qubits) - 1)):\n", " circuit.CX(i, i + 1)\n", " for q, p in term.map.items():\n", " if p == Pauli.X:\n", " circuit.H(q)\n", " elif p == Pauli.Y:\n", " circuit.Vdg(q)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Unitary Coupled Cluster Singles & Doubles ansatz:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def ucc(params):\n", " ansatz = Circuit(4)\n", " # Set initial reference state\n", " ansatz.X(1).X(3)\n", " # Evolve by excitations\n", " for term, coeff in singles_a.items():\n", " add_operator_term(ansatz, term, coeff * params[0])\n", " for term, coeff in singles_b.items():\n", " add_operator_term(ansatz, term, coeff * params[1])\n", " for term, coeff in doubles.items():\n", " add_operator_term(ansatz, term, coeff * params[2])\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is already quite verbose, but `pytket` has a neat shorthand construction for these operator terms using the `PauliExpBox` construction. We can then decompose these into basic gates using the `DecomposeBoxes` compiler pass."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import PauliExpBox\n", "from pytket.passes import DecomposeBoxes"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def add_excitation(circ, term_dict, param):\n", " for term, coeff in term_dict.items():\n", " qubits, paulis = zip(*term.map.items())\n", " pbox = PauliExpBox(paulis, coeff * param)\n", " circ.add_pauliexpbox(pbox, qubits)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["UCC ansatz with syntactic shortcuts:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def ucc(params):\n", " ansatz = Circuit(4)\n", " ansatz.X(1).X(3)\n", " add_excitation(ansatz, singles_a, params[0])\n", " add_excitation(ansatz, singles_b, params[1])\n", " add_excitation(ansatz, doubles, params[2])\n", " DecomposeBoxes().apply(ansatz)\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The objective function can also be simplified using a utility method for constructing the measurement circuits and processing for expectation value calculations. For that, we convert the Hamiltonian to a pytket QubitPauliOperator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["pauli_sym = {\"I\": Pauli.I, \"X\": Pauli.X, \"Y\": Pauli.Y, \"Z\": Pauli.Z}"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qps_from_openfermion(paulis):\n", " \"\"\"Convert OpenFermion tensor of Paulis to pytket QubitPauliString.\"\"\"\n", " qlist = []\n", " plist = []\n", " for q, p in paulis:\n", " qlist.append(Qubit(q))\n", " plist.append(pauli_sym[p])\n", " return QubitPauliString(qlist, plist)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qpo_from_openfermion(openf_op):\n", " \"\"\"Convert OpenFermion QubitOperator to pytket QubitPauliOperator.\"\"\"\n", " tk_op = dict()\n", " for term, coeff in openf_op.terms.items():\n", " string = qps_from_openfermion(term)\n", " tk_op[string] = coeff\n", " return QubitPauliOperator(tk_op)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian_op = qpo_from_openfermion(hamiltonian)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Simplified objective function using utilities:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils.expectations import get_operator_expectation_value"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc(params)\n", " return (\n", " get_operator_expectation_value(circ, hamiltonian_op, backend, n_shots=4000)\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["arg_values = [-3.79002933e-05, 2.42964799e-05, 4.63447157e-01]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["energy = objective(arg_values)\n", "print(energy)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is now the simplest form that this operation can take, but it isn't necessarily the most effective. When we decompose the ansatz circuit into basic gates, it is still very expensive. We can employ some of the circuit simplification passes available in `pytket` to reduce its size and improve fidelity in practice.
\n", "
\n", "A good example is to decompose each `PauliExpBox` into basic gates and then apply `FullPeepholeOptimise`, which defines a compilation strategy utilising all of the simplifications in `pytket` that act locally on small regions of a circuit. We can examine the effectiveness by looking at the number of two-qubit gates before and after simplification, which tends to be a good indicator of fidelity for near-term systems where these gates are often slow and inaccurate."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import OpType\n", "from pytket.passes import FullPeepholeOptimise"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_circuit = ucc(arg_values)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count before\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth before\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["FullPeepholeOptimise().apply(test_circuit)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count after FPO\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth after FPO\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These simplification techniques are very general and are almost always beneficial to apply to a circuit if you want to eliminate local redundancies. But UCC ans\u00e4tze have extra structure that we can exploit further. They are defined entirely out of exponentiated tensors of Pauli matrices, giving the regular structure described by the `PauliExpBox`es. Under many circumstances, it is more efficient to not synthesise these constructions individually, but simultaneously in groups. The `PauliSimp` pass finds the description of a given circuit as a sequence of `PauliExpBox`es and resynthesises them (by default, in groups of commuting terms). This can cause great change in the overall structure and shape of the circuit, enabling the identification and elimination of non-local redundancy."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.passes import PauliSimp"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_circuit = ucc(arg_values)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count before\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth before\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["PauliSimp().apply(test_circuit)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count after PS\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth after PS\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["FullPeepholeOptimise().apply(test_circuit)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count after PS+FPO\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth after PS+FPO\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["To include this into our routines, we can just add the simplification passes to the objective function. The `get_operator_expectation_value` utility handles compiling to meet the requirements of the backend, so we don't have to worry about that here."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function with circuit simplification:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc(params)\n", " PauliSimp().apply(circ)\n", " FullPeepholeOptimise().apply(circ)\n", " return (\n", " get_operator_expectation_value(circ, hamiltonian_op, backend, n_shots=4000)\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These circuit simplification techniques have tried to preserve the exact unitary of the circuit, but there are ways to change the unitary whilst preserving the correctness of the algorithm as a whole.
\n", "
\n", "For example, the excitation terms are generated by trotterisation of the excitation operator, and the order of the terms does not change the unitary in the limit of many trotter steps, so in this sense we are free to sequence the terms how we like and it is sensible to do this in a way that enables efficient synthesis of the circuit. Prioritising collecting terms into commuting sets is a very beneficial heuristic for this and can be performed using the `gen_term_sequence_circuit` method to group the terms together into collections of `PauliExpBox`es and the `GuidedPauliSimp` pass to utilise these sets for synthesis."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.passes import GuidedPauliSimp\n", "from pytket.utils import gen_term_sequence_circuit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def ucc(params):\n", " singles_params = {qps: params[0] * coeff for qps, coeff in singles.items()}\n", " doubles_params = {qps: params[1] * coeff for qps, coeff in doubles.items()}\n", " excitation_op = QubitPauliOperator({**singles_params, **doubles_params})\n", " reference_circ = Circuit(4).X(1).X(3)\n", " ansatz = gen_term_sequence_circuit(excitation_op, reference_circ)\n", " GuidedPauliSimp().apply(ansatz)\n", " FullPeepholeOptimise().apply(ansatz)\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Adding these simplification routines doesn't come for free. Compiling and simplifying the circuit to achieve the best results possible can be a difficult task, which can take some time for the classical computer to perform.
\n", "
\n", "During a VQE run, we will call this objective function many times and run many measurement circuits within each, but the circuits that are run on the quantum computer are almost identical, having the same gate structure but with different gate parameters and measurements. We have already exploited this within the body of the objective function by simplifying the ansatz circuit before we call `get_operator_expectation_value`, so it is only done once per objective calculation rather than once per measurement circuit.
\n", "
\n", "We can go even further by simplifying it once outside of the objective function, and then instantiating the simplified ansatz with the parameter values needed. For this, we will construct the UCC ansatz circuit using symbolic (parametric) gates."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from sympy import symbols"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Symbolic UCC ansatz generation:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["syms = symbols(\"p0 p1 p2\")\n", "singles_a_syms = {qps: syms[0] * coeff for qps, coeff in singles_a.items()}\n", "singles_b_syms = {qps: syms[1] * coeff for qps, coeff in singles_b.items()}\n", "doubles_syms = {qps: syms[2] * coeff for qps, coeff in doubles.items()}\n", "excitation_op = QubitPauliOperator({**singles_a_syms, **singles_b_syms, **doubles_syms})\n", "ucc_ref = Circuit(4).X(1).X(3)\n", "ucc = gen_term_sequence_circuit(excitation_op, ucc_ref)\n", "GuidedPauliSimp().apply(ucc)\n", "FullPeepholeOptimise().apply(ucc)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function using the symbolic ansatz:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc.copy()\n", " sym_map = dict(zip(syms, params))\n", " circ.symbol_substitution(sym_map)\n", " return (\n", " get_operator_expectation_value(circ, hamiltonian_op, backend, n_shots=4000)\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We have now got some very good use of `pytket` for simplifying each individual circuit used in our experiment and for minimising the amount of time spent compiling, but there is still more we can do in terms of reducing the amount of work the quantum computer has to do. Currently, each (non-trivial) term in our measurement hamiltonian is measured by a different circuit within each expectation value calculation. Measurement reduction techniques exist for identifying when these observables commute and hence can be simultaneously measured, reducing the number of circuits required for the full expectation value calculation.
\n", "
\n", "This is built in to the `get_operator_expectation_value` method and can be applied by specifying a way to partition the measuremrnt terms. `PauliPartitionStrat.CommutingSets` can greatly reduce the number of measurement circuits by combining any number of terms that mutually commute. However, this involves potentially adding an arbitrary Clifford circuit to change the basis of the measurements which can be costly on NISQ devices, so `PauliPartitionStrat.NonConflictingSets` trades off some of the reduction in circuit number to guarantee that only single-qubit gates are introduced."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.partition import PauliPartitionStrat"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function using measurement reduction:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc.copy()\n", " sym_map = dict(zip(syms, params))\n", " circ.symbol_substitution(sym_map)\n", " return (\n", " get_operator_expectation_value(\n", " circ,\n", " operator,\n", " backend,\n", " n_shots=4000,\n", " partition_strat=PauliPartitionStrat.CommutingSets,\n", " )\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "markdown", "metadata": {}, "source": ["At this point, we have completely transformed how our VQE objective function works, improving its resilience to noise, cutting the number of circuits run, and maintaining fast runtimes. In doing this, we have explored a number of the features `pytket` offers that are beneficial to VQE and the UCC method:
\n", "- high-level syntactic constructs for evolution operators;
\n", "- utility methods for easy expectation value calculations;
\n", "- both generic and domain-specific circuit simplification methods;
\n", "- symbolic circuit compilation;
\n", "- measurement reduction for expectation value calculations."]}, {"cell_type": "markdown", "metadata": {}, "source": ["For the sake of completeness, the following gives the full code for the final solution, including passing the objective function to a classical optimiser to find the ground state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import openfermion as of\n", "from scipy.optimize import minimize\n", "from sympy import symbols"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import AerBackend\n", "from pytket.circuit import Circuit, Qubit\n", "from pytket.partition import PauliPartitionStrat\n", "from pytket.passes import GuidedPauliSimp, FullPeepholeOptimise\n", "from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.utils import get_operator_expectation_value, gen_term_sequence_circuit\n", "from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Obtain electronic Hamiltonian:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian = (\n", " -0.8153001706270075 * of.QubitOperator(\"\")\n", " + 0.16988452027940318 * of.QubitOperator(\"Z0\")\n", " + -0.21886306781219608 * of.QubitOperator(\"Z1\")\n", " + 0.16988452027940323 * of.QubitOperator(\"Z2\")\n", " + -0.2188630678121961 * of.QubitOperator(\"Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z0 Z1\")\n", " + 0.16821198673715723 * of.QubitOperator(\"Z0 Z2\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z0 Z3\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z1 Z2\")\n", " + 0.1739537877649417 * of.QubitOperator(\"Z1 Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z2 Z3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 Y2 Y3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 Y2 Y3\")\n", ")\n", "nuclear_repulsion_energy = 0.70556961456"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["pauli_sym = {\"I\": Pauli.I, \"X\": Pauli.X, \"Y\": Pauli.Y, \"Z\": Pauli.Z}"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qps_from_openfermion(paulis):\n", " \"\"\"Convert OpenFermion tensor of Paulis to pytket QubitPauliString.\"\"\"\n", " qlist = []\n", " plist = []\n", " for q, p in paulis:\n", " qlist.append(Qubit(q))\n", " plist.append(pauli_sym[p])\n", " return QubitPauliString(qlist, plist)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qpo_from_openfermion(openf_op):\n", " \"\"\"Convert OpenFermion QubitOperator to pytket QubitPauliOperator.\"\"\"\n", " tk_op = dict()\n", " for term, coeff in openf_op.terms.items():\n", " string = qps_from_openfermion(term)\n", " tk_op[string] = coeff\n", " return QubitPauliOperator(tk_op)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian_op = qpo_from_openfermion(hamiltonian)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Obtain terms for single and double excitations:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["q = [Qubit(i) for i in range(4)]\n", "xyii = QubitPauliString([q[0], q[1]], [Pauli.X, Pauli.Y])\n", "yxii = QubitPauliString([q[0], q[1]], [Pauli.Y, Pauli.X])\n", "iixy = QubitPauliString([q[2], q[3]], [Pauli.X, Pauli.Y])\n", "iiyx = QubitPauliString([q[2], q[3]], [Pauli.Y, Pauli.X])\n", "xxxy = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.X, Pauli.Y])\n", "xxyx = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.Y, Pauli.X])\n", "xyxx = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.X, Pauli.X])\n", "yxxx = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.X, Pauli.X])\n", "yyyx = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.Y, Pauli.X])\n", "yyxy = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.X, Pauli.Y])\n", "yxyy = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.Y, Pauli.Y])\n", "xyyy = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.Y, Pauli.Y])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Symbolic UCC ansatz generation:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["syms = symbols(\"p0 p1 p2\")\n", "singles_syms = {xyii: syms[0], yxii: -syms[0], iixy: syms[1], iiyx: -syms[1]}\n", "doubles_syms = {\n", " xxxy: 0.25 * syms[2],\n", " xxyx: -0.25 * syms[2],\n", " xyxx: 0.25 * syms[2],\n", " yxxx: -0.25 * syms[2],\n", " yyyx: -0.25 * syms[2],\n", " yyxy: 0.25 * syms[2],\n", " yxyy: -0.25 * syms[2],\n", " xyyy: 0.25 * syms[2],\n", "}\n", "excitation_op = QubitPauliOperator({**singles_syms, **doubles_syms})\n", "ucc_ref = Circuit(4).X(0).X(2)\n", "ucc = gen_term_sequence_circuit(excitation_op, ucc_ref)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Circuit simplification:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["GuidedPauliSimp().apply(ucc)\n", "FullPeepholeOptimise().apply(ucc)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Connect to a simulator/device:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc.copy()\n", " sym_map = dict(zip(syms, params))\n", " circ.symbol_substitution(sym_map)\n", " return (\n", " get_operator_expectation_value(\n", " circ,\n", " hamiltonian_op,\n", " backend,\n", " n_shots=4000,\n", " partition_strat=PauliPartitionStrat.CommutingSets,\n", " )\n", " + nuclear_repulsion_energy\n", " ).real"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Optimise against the objective function:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["initial_params = [1e-4, 1e-4, 4e-1]\n", "# #result = minimize(objective, initial_params, method=\"Nelder-Mead\")\n", "# #print(\"Final parameter values\", result.x)\n", "# #print(\"Final energy value\", result.fun)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Exercises:
\n", "- Replace the `get_operator_expectation_value` call with its implementation and use this to pull the analysis for measurement reduction outside of the objective function, so our circuits can be fully determined and compiled once. This means that the `symbol_substitution` method will need to be applied to each measurement circuit instead of just the state preparation circuit.
\n", "- Use the `SpamCorrecter` class to add some mitigation of the measurement errors. Start by running the characterisation circuits first, before your main VQE loop, then apply the mitigation to each of the circuits run within the objective function.
\n", "- Change the `backend` by passing in a `Qiskit` `NoiseModel` to simulate a noisy device. Compare the accuracy of the objective function both with and without the circuit simplification. Try running a classical optimiser over the objective function and compare the convergence rates with different noise models. If you have access to a QPU, try changing the `backend` to connect to that and compare the results to the simulator."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file From c65288c4a35b7a1fc168ba206eab71a3bc6d40e8 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 26 Oct 2023 17:48:41 +0100 Subject: [PATCH 15/51] consistent capitalisation --- examples/_toc.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/examples/_toc.yml b/examples/_toc.yml index efa98413..b02b0748 100644 --- a/examples/_toc.yml +++ b/examples/_toc.yml @@ -9,14 +9,14 @@ parts: - file: circuit_generation_example - file: circuit_analysis_example - file: conditional_gate_example - - caption: TKET backends + - caption: TKET Backends chapters: - file: backends_example - file: comparing_simulators - file: Forest_portability_example - file: creating_backends - file: qiskit_integration - - caption: Circuit compilation + - caption: Circuit Compilation chapters: - file: compilation_example - file: symbolics_example From 69a0bb1e0c1b66d4dc4dd6ce9e6d75e37515e686 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 26 Oct 2023 17:51:47 +0100 Subject: [PATCH 16/51] TKET Backends -> Backends --- examples/_toc.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/examples/_toc.yml b/examples/_toc.yml index b02b0748..c5c32481 100644 --- a/examples/_toc.yml +++ b/examples/_toc.yml @@ -9,7 +9,7 @@ parts: - file: circuit_generation_example - file: circuit_analysis_example - file: conditional_gate_example - - caption: TKET Backends + - caption: Backends chapters: - file: backends_example - file: comparing_simulators From eee13a22cb1af21b25f9670817059ce6fdb1abd9 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 26 Oct 2023 17:52:47 +0100 Subject: [PATCH 17/51] fix heading in comparing simulators --- examples/comparing_simulators.ipynb | 2 +- examples/python/comparing_simulators.py | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/examples/comparing_simulators.ipynb b/examples/comparing_simulators.ipynb index 60405fcd..0931a019 100644 --- a/examples/comparing_simulators.ipynb +++ b/examples/comparing_simulators.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Comparison of the simulators available through tket"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In this tutorial, we will focus on:
\n", "- exploring the wide array of simulators available through the extension modules for `pytket`;
\n", "- comparing their unique features and capabilities."]}, {"cell_type": "markdown", "metadata": {}, "source": ["This example assumes the reader is familiar with the basics of circuit construction and evaluation.
\n", "
\n", "To run every option in this example, you will need `pytket`, `pytket-qiskit`, `pytket-pyquil`, `pytket-qsharp`, `pytket-qulacs`, and `pytket-projectq`.
\n", "
\n", "With the number of simulator `Backend`s available across the `pytket` extension modules, we are often asked why to use one over another. Surely, any two simulators are equivalent if they are able to sample the circuits in the same way, right? Not quite. In this notebook we go through each of the simulators in turn and describe what sets them apart from others and how to make use of any unique features.
\n", "
\n", "But first, to demonstrate the significant overlap in functionality, we'll just give some examples of common usage for different types of backends."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Sampling simulator usage"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.extensions.qiskit import AerBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Define a circuit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3, 3)\n", "c.Ry(0.7, 0)\n", "c.CX(0, 1)\n", "c.X(2)\n", "c.measure_all()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run on the backend:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()\n", "c = backend.get_compiled_circuit(c)\n", "handle = backend.process_circuit(c, n_shots=2000)\n", "counts = backend.get_result(handle).get_counts()\n", "print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Statevector simulator usage"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.extensions.qiskit import AerStateBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Build a quantum state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3)\n", "c.H(0).CX(0, 1)\n", "c.Rz(0.3, 0)\n", "c.Rz(-0.3, 1)\n", "c.Ry(0.8, 2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Examine the statevector:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerStateBackend()\n", "c = backend.get_compiled_circuit(c)\n", "handle = backend.process_circuit(c)\n", "state = backend.get_result(handle).get_state()\n", "print(state)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Expectation value usage"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit, Qubit\n", "from pytket.extensions.qiskit import AerBackend, AerStateBackend\n", "from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Build a quantum state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3)\n", "c.H(0).CX(0, 1)\n", "c.Rz(0.3, 0)\n", "c.Rz(-0.3, 1)\n", "c.Ry(0.8, 2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Define the measurement operator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["xxi = QubitPauliString({Qubit(0): Pauli.X, Qubit(1): Pauli.X})\n", "zzz = QubitPauliString({Qubit(0): Pauli.Z, Qubit(1): Pauli.Z, Qubit(2): Pauli.Z})\n", "op = QubitPauliOperator({xxi: -1.8, zzz: 0.7})"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run on the backend:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()\n", "c = backend.get_compiled_circuit(c)\n", "exp = backend.get_operator_expectation_value(c, op)\n", "print(exp)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `AerBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["`AerBackend` wraps up the `qasm_simulator` from the Qiskit Aer package. It supports an extremely flexible set of circuits and uses many effective simulation methods making it a great all-purpose sampling simulator.
\n", "
\n", "Unique features:
\n", "- supports mid-circuit measurement and OpenQASM-style conditional gates;
\n", "- encompasses a variety of underlying simulation methods and automatically selects the best one for each circuit (including statevector, density matrix, (extended) stabilizer and matrix product state);
\n", "- can be provided with a `qiskit.providers.Aer.noise.NoiseModel` on instantiation to perform a noisy simulation."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Useful features:
\n", "- support for fast expectation value calculations according to `QubitPauliString`s or `QubitPauliOperator`s."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.extensions.qiskit import AerBackend\n", "from itertools import combinations\n", "from qiskit.providers.aer.noise import NoiseModel, depolarizing_error"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Quantum teleportation circuit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit()\n", "alice = c.add_q_register(\"a\", 2)\n", "bob = c.add_q_register(\"b\", 1)\n", "data = c.add_c_register(\"d\", 2)\n", "final = c.add_c_register(\"f\", 1)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Start in an interesting state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.Rx(0.3, alice[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Set up a Bell state between Alice and Bob:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.H(alice[1]).CX(alice[1], bob[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Measure Alice's qubits in the Bell basis:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.CX(alice[0], alice[1]).H(alice[0])\n", "c.Measure(alice[0], data[0])\n", "c.Measure(alice[1], data[1])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Correct Bob's qubit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.X(bob[0], condition_bits=[data[0], data[1]], condition_value=1)\n", "c.X(bob[0], condition_bits=[data[0], data[1]], condition_value=3)\n", "c.Z(bob[0], condition_bits=[data[0], data[1]], condition_value=2)\n", "c.Z(bob[0], condition_bits=[data[0], data[1]], condition_value=3)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Measure Bob's qubit to observe the interesting state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.Measure(bob[0], final[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Set up a noisy simulator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["model = NoiseModel()\n", "dep_err = depolarizing_error(0.04, 2)\n", "for i, j in combinations(range(3), r=2):\n", " model.add_quantum_error(dep_err, [\"cx\"], [i, j])\n", " model.add_quantum_error(dep_err, [\"cx\"], [j, i])\n", "backend = AerBackend(noise_model=model)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run circuit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = backend.get_compiled_circuit(c)\n", "handle = backend.process_circuit(c, n_shots=2000)\n", "result = backend.get_result(handle)\n", "counts = result.get_counts([final[0]])\n", "print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `AerStateBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["`AerStateBackend` provides access to Qiskit Aer's `statevector_simulator`. It supports a similarly large gate set and has competitive speed for statevector simulations.
\n", "
\n", "Useful features:
\n", "- no dependency on external executables, making it easy to install and run on any computer;
\n", "- support for fast expectation value calculations according to `QubitPauliString`s or `QubitPauliOperator`s."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `AerUnitaryBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Finishing the set of simulators from Qiskit Aer, `AerUnitaryBackend` captures the `unitary_simulator`, allowing for the entire unitary of a pure quantum process to be calculated. This is especially useful for testing small subcircuits that will be used many times in a larger computation.
\n", "
\n", "Unique features:
\n", "- provides the full unitary matrix for a pure quantum circuit."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.extensions.qiskit import AerUnitaryBackend\n", "from pytket.predicates import NoClassicalControlPredicate"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Define a simple quantum incrementer:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3)\n", "c.CCX(2, 1, 0)\n", "c.CX(2, 1)\n", "c.X(2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Examine the unitary:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerUnitaryBackend()\n", "c = backend.get_compiled_circuit(c)\n", "result = backend.run_circuit(c)\n", "unitary = result.get_unitary()\n", "print(unitary.round(1).real)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `ForestBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Whilst it can, with suitable credentials, be used to access the Rigetti QPUs, the `ForestBackend` also features a simulator mode which turns it into a noiseless sampling simulator that matches the constraints of the simulated device (e.g. the same gate set, restricted connectivity, measurement model, etc.). This is useful when playing around with custom compilation strategies to ensure that your final circuits are suitable to run on the device and for checking that your overall program works fine before you invest in reserving a QPU.
\n", "
\n", "Unique features:
\n", "- faithful recreation of the circuit constraints of Rigetti QPUs."]}, {"cell_type": "markdown", "metadata": {}, "source": ["If trying to use the `ForestBackend` locally (i.e. not on a Rigetti QMI), you will need to have `quilc` and `qvm` running as separate processes in server mode. One easy way of doing this is with `docker` (see the `quilc` and `qvm` documentation for alternative methods of running them):
\n", "`docker run --rm -it -p 5555:5555 rigetti/quilc -R`
\n", "`docker run --rm -it -p 5000:5000 rigetti/qvm -S`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `ForestStateBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The Rigetti `pyquil` package also provides the `WavefunctionSimulator`, which we present as the `ForestStateBackend`. Functionally, it is very similar to the `AerStateBackend` so can be used interchangeably. It does require that `quilc` and `qvm` are running as separate processes when not running on a Rigetti QMI.
\n", "
\n", "Useful features:
\n", "- support for fast expectation value calculations according to `QubitPauliString`s or Hermitian `QubitPauliOperator`s."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `QsharpSimulatorBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The `QsharpSimulatorBackend` is another basic sampling simulator that is interchangeable with others, using the Microsoft QDK simulator. Note that the `pytket-qsharp` package is dependent on the `dotnet` SDK and `iqsharp` tool. Please consult the `pytket-qsharp` installation instructions for recommendations."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `QsharpToffoliSimulatorBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Toffoli circuits form a strict fragment of quantum circuits and can be efficiently simulated. The `QsharpToffoliSimulatorBackend` can only operate on these circuits, but scales much better with system size than regular simulators.
\n", "
\n", "Unique features:
\n", "- efficient simulation of Toffoli circuits."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.extensions.qsharp import QsharpToffoliSimulatorBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Define a circuit - start in a basis state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3)\n", "c.X(0).X(2)\n", "# Define a circuit - incrementer\n", "c.CCX(2, 1, 0)\n", "c.CX(2, 1)\n", "c.X(2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run on the backend:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = QsharpToffoliSimulatorBackend()\n", "c = backend.get_compiled_circuit(c)\n", "handle = backend.process_circuit(c, n_shots=10)\n", "counts = backend.get_result(handle).get_counts()\n", "print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `QsharpEstimatorBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The `QsharpEstimatorBackend` is not strictly a simulator, as it doesn't model the state of the quantum system and try to identify the final state, but instead analyses the circuit to estimate the required resources to run it. It does not support any of the regular outcome types (e.g. shots, counts, statevector), just the summary of the estimated resources.
\n", "
\n", "Unique features:
\n", "- estimates resources to perform the circuit, without actually simulating/running it."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit"]}, {"cell_type": "markdown", "metadata": {}, "source": ["from pytket.extensions.qsharp import QsharpEstimatorBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Define a circuit - start in a basis state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3)\n", "c.X(0).X(2)\n", "# Define a circuit - incrementer\n", "c.CCX(2, 1, 0)\n", "c.CX(2, 1)\n", "c.X(2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run on the backend:"]}, {"cell_type": "markdown", "metadata": {}, "source": ["(disabled because of https://github.com/CQCL/pytket-qsharp/issues/37)
\n", "backend = QsharpEstimatorBackend()
\n", "c = backend.get_compiled_circuit(c)
\n", "handle = backend.process_circuit(c, n_shots=10)
\n", "resources = backend.get_resources(handle)
\n", "print(resources)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `QulacsBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The `QulacsBackend` is an all-purpose simulator with both sampling and statevector modes, using the basic CPU simulator from Qulacs.
\n", "
\n", "Unique features:
\n", "- supports both sampling (shots/counts) and complete statevector outputs."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Useful features:
\n", "- support for fast expectation value calculations according to `QubitPauliString`s or Hermitian `QubitPauliOperator`s."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `QulacsGPUBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["If the GPU version of Qulacs is installed, the `QulacsGPUBackend` will use that to benefit from even faster speeds. It is very easy to get started with using a GPU, as it only requires a CUDA installation and the `qulacs-gpu` package from `pip`. Functionally, it is identical to the `QulacsBackend`, but potentially faster if you have GPU resources available.
\n", "
\n", "Unique features:
\n", "- GPU support for very fast simulation."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `ProjectQBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["ProjectQ is a popular quantum circuit simulator, thanks to its availability and ease of use. It provides a similar level of performance and features to `AerStateBackend`.
\n", "
\n", "Useful features:
\n", "- support for fast expectation value calculations according to `QubitPauliString`s or Hermitian `QubitPauliOperator`s."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Comparison of the simulators available through TKET"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In this tutorial, we will focus on:
\n", "- exploring the wide array of simulators available through the extension modules for `pytket`;
\n", "- comparing their unique features and capabilities."]}, {"cell_type": "markdown", "metadata": {}, "source": ["This example assumes the reader is familiar with the basics of circuit construction and evaluation.
\n", "
\n", "To run every option in this example, you will need `pytket`, `pytket-qiskit`, `pytket-pyquil`, `pytket-qsharp`, `pytket-qulacs`, and `pytket-projectq`.
\n", "
\n", "With the number of simulator `Backend`s available across the `pytket` extension modules, we are often asked why to use one over another. Surely, any two simulators are equivalent if they are able to sample the circuits in the same way, right? Not quite. In this notebook we go through each of the simulators in turn and describe what sets them apart from others and how to make use of any unique features.
\n", "
\n", "But first, to demonstrate the significant overlap in functionality, we'll just give some examples of common usage for different types of backends."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Sampling simulator usage"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.extensions.qiskit import AerBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Define a circuit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3, 3)\n", "c.Ry(0.7, 0)\n", "c.CX(0, 1)\n", "c.X(2)\n", "c.measure_all()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run on the backend:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()\n", "c = backend.get_compiled_circuit(c)\n", "handle = backend.process_circuit(c, n_shots=2000)\n", "counts = backend.get_result(handle).get_counts()\n", "print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Statevector simulator usage"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.extensions.qiskit import AerStateBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Build a quantum state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3)\n", "c.H(0).CX(0, 1)\n", "c.Rz(0.3, 0)\n", "c.Rz(-0.3, 1)\n", "c.Ry(0.8, 2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Examine the statevector:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerStateBackend()\n", "c = backend.get_compiled_circuit(c)\n", "handle = backend.process_circuit(c)\n", "state = backend.get_result(handle).get_state()\n", "print(state)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Expectation value usage"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit, Qubit\n", "from pytket.extensions.qiskit import AerBackend, AerStateBackend\n", "from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Build a quantum state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3)\n", "c.H(0).CX(0, 1)\n", "c.Rz(0.3, 0)\n", "c.Rz(-0.3, 1)\n", "c.Ry(0.8, 2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Define the measurement operator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["xxi = QubitPauliString({Qubit(0): Pauli.X, Qubit(1): Pauli.X})\n", "zzz = QubitPauliString({Qubit(0): Pauli.Z, Qubit(1): Pauli.Z, Qubit(2): Pauli.Z})\n", "op = QubitPauliOperator({xxi: -1.8, zzz: 0.7})"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run on the backend:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()\n", "c = backend.get_compiled_circuit(c)\n", "exp = backend.get_operator_expectation_value(c, op)\n", "print(exp)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `AerBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["`AerBackend` wraps up the `qasm_simulator` from the Qiskit Aer package. It supports an extremely flexible set of circuits and uses many effective simulation methods making it a great all-purpose sampling simulator.
\n", "
\n", "Unique features:
\n", "- supports mid-circuit measurement and OpenQASM-style conditional gates;
\n", "- encompasses a variety of underlying simulation methods and automatically selects the best one for each circuit (including statevector, density matrix, (extended) stabilizer and matrix product state);
\n", "- can be provided with a `qiskit.providers.Aer.noise.NoiseModel` on instantiation to perform a noisy simulation."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Useful features:
\n", "- support for fast expectation value calculations according to `QubitPauliString`s or `QubitPauliOperator`s."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.extensions.qiskit import AerBackend\n", "from itertools import combinations\n", "from qiskit.providers.aer.noise import NoiseModel, depolarizing_error"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Quantum teleportation circuit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit()\n", "alice = c.add_q_register(\"a\", 2)\n", "bob = c.add_q_register(\"b\", 1)\n", "data = c.add_c_register(\"d\", 2)\n", "final = c.add_c_register(\"f\", 1)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Start in an interesting state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.Rx(0.3, alice[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Set up a Bell state between Alice and Bob:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.H(alice[1]).CX(alice[1], bob[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Measure Alice's qubits in the Bell basis:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.CX(alice[0], alice[1]).H(alice[0])\n", "c.Measure(alice[0], data[0])\n", "c.Measure(alice[1], data[1])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Correct Bob's qubit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.X(bob[0], condition_bits=[data[0], data[1]], condition_value=1)\n", "c.X(bob[0], condition_bits=[data[0], data[1]], condition_value=3)\n", "c.Z(bob[0], condition_bits=[data[0], data[1]], condition_value=2)\n", "c.Z(bob[0], condition_bits=[data[0], data[1]], condition_value=3)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Measure Bob's qubit to observe the interesting state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c.Measure(bob[0], final[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Set up a noisy simulator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["model = NoiseModel()\n", "dep_err = depolarizing_error(0.04, 2)\n", "for i, j in combinations(range(3), r=2):\n", " model.add_quantum_error(dep_err, [\"cx\"], [i, j])\n", " model.add_quantum_error(dep_err, [\"cx\"], [j, i])\n", "backend = AerBackend(noise_model=model)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run circuit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = backend.get_compiled_circuit(c)\n", "handle = backend.process_circuit(c, n_shots=2000)\n", "result = backend.get_result(handle)\n", "counts = result.get_counts([final[0]])\n", "print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `AerStateBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["`AerStateBackend` provides access to Qiskit Aer's `statevector_simulator`. It supports a similarly large gate set and has competitive speed for statevector simulations.
\n", "
\n", "Useful features:
\n", "- no dependency on external executables, making it easy to install and run on any computer;
\n", "- support for fast expectation value calculations according to `QubitPauliString`s or `QubitPauliOperator`s."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `AerUnitaryBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Finishing the set of simulators from Qiskit Aer, `AerUnitaryBackend` captures the `unitary_simulator`, allowing for the entire unitary of a pure quantum process to be calculated. This is especially useful for testing small subcircuits that will be used many times in a larger computation.
\n", "
\n", "Unique features:
\n", "- provides the full unitary matrix for a pure quantum circuit."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.extensions.qiskit import AerUnitaryBackend\n", "from pytket.predicates import NoClassicalControlPredicate"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Define a simple quantum incrementer:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3)\n", "c.CCX(2, 1, 0)\n", "c.CX(2, 1)\n", "c.X(2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Examine the unitary:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerUnitaryBackend()\n", "c = backend.get_compiled_circuit(c)\n", "result = backend.run_circuit(c)\n", "unitary = result.get_unitary()\n", "print(unitary.round(1).real)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `ForestBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Whilst it can, with suitable credentials, be used to access the Rigetti QPUs, the `ForestBackend` also features a simulator mode which turns it into a noiseless sampling simulator that matches the constraints of the simulated device (e.g. the same gate set, restricted connectivity, measurement model, etc.). This is useful when playing around with custom compilation strategies to ensure that your final circuits are suitable to run on the device and for checking that your overall program works fine before you invest in reserving a QPU.
\n", "
\n", "Unique features:
\n", "- faithful recreation of the circuit constraints of Rigetti QPUs."]}, {"cell_type": "markdown", "metadata": {}, "source": ["If trying to use the `ForestBackend` locally (i.e. not on a Rigetti QMI), you will need to have `quilc` and `qvm` running as separate processes in server mode. One easy way of doing this is with `docker` (see the `quilc` and `qvm` documentation for alternative methods of running them):
\n", "`docker run --rm -it -p 5555:5555 rigetti/quilc -R`
\n", "`docker run --rm -it -p 5000:5000 rigetti/qvm -S`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `ForestStateBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The Rigetti `pyquil` package also provides the `WavefunctionSimulator`, which we present as the `ForestStateBackend`. Functionally, it is very similar to the `AerStateBackend` so can be used interchangeably. It does require that `quilc` and `qvm` are running as separate processes when not running on a Rigetti QMI.
\n", "
\n", "Useful features:
\n", "- support for fast expectation value calculations according to `QubitPauliString`s or Hermitian `QubitPauliOperator`s."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `QsharpSimulatorBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The `QsharpSimulatorBackend` is another basic sampling simulator that is interchangeable with others, using the Microsoft QDK simulator. Note that the `pytket-qsharp` package is dependent on the `dotnet` SDK and `iqsharp` tool. Please consult the `pytket-qsharp` installation instructions for recommendations."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `QsharpToffoliSimulatorBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Toffoli circuits form a strict fragment of quantum circuits and can be efficiently simulated. The `QsharpToffoliSimulatorBackend` can only operate on these circuits, but scales much better with system size than regular simulators.
\n", "
\n", "Unique features:
\n", "- efficient simulation of Toffoli circuits."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.extensions.qsharp import QsharpToffoliSimulatorBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Define a circuit - start in a basis state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3)\n", "c.X(0).X(2)\n", "# Define a circuit - incrementer\n", "c.CCX(2, 1, 0)\n", "c.CX(2, 1)\n", "c.X(2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run on the backend:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = QsharpToffoliSimulatorBackend()\n", "c = backend.get_compiled_circuit(c)\n", "handle = backend.process_circuit(c, n_shots=10)\n", "counts = backend.get_result(handle).get_counts()\n", "print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `QsharpEstimatorBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The `QsharpEstimatorBackend` is not strictly a simulator, as it doesn't model the state of the quantum system and try to identify the final state, but instead analyses the circuit to estimate the required resources to run it. It does not support any of the regular outcome types (e.g. shots, counts, statevector), just the summary of the estimated resources.
\n", "
\n", "Unique features:
\n", "- estimates resources to perform the circuit, without actually simulating/running it."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit"]}, {"cell_type": "markdown", "metadata": {}, "source": ["from pytket.extensions.qsharp import QsharpEstimatorBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Define a circuit - start in a basis state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3)\n", "c.X(0).X(2)\n", "# Define a circuit - incrementer\n", "c.CCX(2, 1, 0)\n", "c.CX(2, 1)\n", "c.X(2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run on the backend:"]}, {"cell_type": "markdown", "metadata": {}, "source": ["(disabled because of https://github.com/CQCL/pytket-qsharp/issues/37)
\n", "backend = QsharpEstimatorBackend()
\n", "c = backend.get_compiled_circuit(c)
\n", "handle = backend.process_circuit(c, n_shots=10)
\n", "resources = backend.get_resources(handle)
\n", "print(resources)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `QulacsBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The `QulacsBackend` is an all-purpose simulator with both sampling and statevector modes, using the basic CPU simulator from Qulacs.
\n", "
\n", "Unique features:
\n", "- supports both sampling (shots/counts) and complete statevector outputs."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Useful features:
\n", "- support for fast expectation value calculations according to `QubitPauliString`s or Hermitian `QubitPauliOperator`s."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `QulacsGPUBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["If the GPU version of Qulacs is installed, the `QulacsGPUBackend` will use that to benefit from even faster speeds. It is very easy to get started with using a GPU, as it only requires a CUDA installation and the `qulacs-gpu` package from `pip`. Functionally, it is identical to the `QulacsBackend`, but potentially faster if you have GPU resources available.
\n", "
\n", "Unique features:
\n", "- GPU support for very fast simulation."]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `ProjectQBackend`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["ProjectQ is a popular quantum circuit simulator, thanks to its availability and ease of use. It provides a similar level of performance and features to `AerStateBackend`.
\n", "
\n", "Useful features:
\n", "- support for fast expectation value calculations according to `QubitPauliString`s or Hermitian `QubitPauliOperator`s."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file diff --git a/examples/python/comparing_simulators.py b/examples/python/comparing_simulators.py index b07b9352..221b3889 100644 --- a/examples/python/comparing_simulators.py +++ b/examples/python/comparing_simulators.py @@ -1,4 +1,4 @@ -# # Comparison of the simulators available through tket +# # Comparison of the simulators available through TKET # In this tutorial, we will focus on: # - exploring the wide array of simulators available through the extension modules for `pytket`; From a70efb75c9d852bea59da397fbc86c9823ddae86 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 26 Oct 2023 18:01:03 +0100 Subject: [PATCH 18/51] change some more captialisations --- examples/Forest_portability_example.ipynb | 2 +- examples/backends_example.ipynb | 2 +- examples/conditional_gate_example.ipynb | 2 +- examples/entanglement_swapping.ipynb | 2 +- examples/expectation_value_example.ipynb | 2 +- examples/mapping_example.ipynb | 2 +- examples/python/Forest_portability_example.py | 2 +- examples/python/backends_example.py | 2 +- examples/python/conditional_gate_example.py | 2 +- examples/python/entanglement_swapping.py | 2 +- examples/python/expectation_value_example.py | 2 +- examples/python/mapping_example.py | 2 +- examples/python/pytket-qujax-classification.py | 2 +- examples/pytket-qujax-classification.ipynb | 2 +- 14 files changed, 14 insertions(+), 14 deletions(-) diff --git a/examples/Forest_portability_example.ipynb b/examples/Forest_portability_example.ipynb index 90e49378..fea497a0 100644 --- a/examples/Forest_portability_example.ipynb +++ b/examples/Forest_portability_example.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Code Portability and Intro to Forest"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The quantum hardware landscape is incredibly competitive and rapidly changing. Many full-stack quantum software platforms lock users into them in order to use the associated devices and simulators. This notebook demonstrates how `pytket` can free up your existing high-level code to be used on devices from other providers. We will take a state-preparation and evolution circuit generated using `qiskit`, and enable it to be run on several Rigetti backends.
\n", "
\n", "To use a real hardware device, this notebook should be run from a Rigetti QMI instance. Look [here](https://www.rigetti.com/qcs/docs/intro-to-qcs) for information on how to set this up. Otherwise, make sure you have QuilC and QVM running in server mode. You will need to have `pytket`, `pytket_pyquil`, and `pytket_qiskit` installed, which are all available from PyPI."]}, {"cell_type": "markdown", "metadata": {}, "source": ["We will start by using `qiskit` to build a random initial state over some qubits. (We remove the initial \"reset\" gates from the circuit since these are not recognized by the Forest backends, which assume an all-zero initial state.)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from qiskit import QuantumCircuit\n", "from qiskit.quantum_info.states.random import random_statevector"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["n_qubits = 3\n", "state = random_statevector((1 << n_qubits, 1)).data\n", "state_prep_circ = QuantumCircuit(n_qubits)\n", "state_prep_circ.initialize(state)\n", "state_prep_circ = state_prep_circ.decompose().decompose()\n", "state_prep_circ.data = [\n", " datum for datum in state_prep_circ.data if datum[0].name != \"reset\"\n", "]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(state_prep_circ)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can now evolve this state under an operator for a given duration."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from qiskit.opflow import PauliTrotterEvolution\n", "from qiskit.opflow.primitive_ops import PauliSumOp\n", "from qiskit.quantum_info import Pauli"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["duration = 1.2\n", "op = PauliSumOp.from_list([(\"XXI\", 0.3), (\"YYI\", 0.5), (\"ZZZ\", -0.4)])\n", "evolved_op = (duration * op).exp_i()\n", "evolution_circ = PauliTrotterEvolution(reps=1).convert(evolved_op).to_circuit()\n", "print(evolution_circ)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["for op in evolution_circ:\n", " state_prep_circ.append(op)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now that we have a circuit, `pytket` can take this and start operating on it directly. For example, we can apply some basic compilation passes to simplify it."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import qiskit_to_tk"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["tk_circ = qiskit_to_tk(state_prep_circ)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.passes import (\n", " SequencePass,\n", " CliffordSimp,\n", " DecomposeBoxes,\n", " KAKDecomposition,\n", " SynthesiseTket,\n", ")"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["DecomposeBoxes().apply(tk_circ)\n", "optimise = SequencePass([KAKDecomposition(), CliffordSimp(False), SynthesiseTket()])\n", "optimise.apply(tk_circ)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Display the optimised circuit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit.display import render_circuit_jupyter"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["render_circuit_jupyter(tk_circ)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The Backends in `pytket` abstract away the differences between different devices and simulators as much as possible, allowing painless switching between them. The `pytket_pyquil` package provides two Backends: `ForestBackend` encapsulates both running on physical devices via Rigetti QCS and simulating those devices on the QVM, and `ForestStateBackend` acts as a wrapper to the pyQuil Wavefunction Simulator.
\n", "
\n", "Both of these still have a few restrictions on the circuits that can be run. Each only supports a subset of the gate types available in `pytket`, and a real device or associated simulation will have restricted qubit connectivity. The Backend objects will contain a default compilation pass that will statisfy these constraints as much as possible, with minimal or no optimisation.
\n", "
\n", "The `ForestStateBackend` will allow us to view the full statevector (wavefunction) expected from a perfect execution of the circuit."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.pyquil import ForestStateBackend"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["state_backend = ForestStateBackend()\n", "tk_circ = state_backend.get_compiled_circuit(tk_circ)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["handle = state_backend.process_circuit(tk_circ)\n", "state = state_backend.get_result(handle).get_state()\n", "print(state)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["For users who are familiar with the Forest SDK, the association of qubits to indices of bitstrings (and consequently the ordering of statevectors) used by default in `pytket` Backends differs from that described in the [Forest docs](http://docs.rigetti.com/en/stable/wavefunction_simulator.html#multi-qubit-basis-enumeration). You can recover the ordering used by the Forest systems with `BackendResult.get_state(tk_circ, basis:pytket.BasisOrder.dlo)` (see our docs on the `BasisOrder` enum for more details)."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Connecting to real devices works very similarly. Instead of obtaining the full statevector, we are only able to measure the quantum state and sample from the resulting distribution. Beyond that, the process is pretty much the same.
\n", "
\n", "The following shows how to run the circuit on the \"9q-square\" lattice. The `as_qvm` switch on the `get_qc` method will switch between connecting to the real Aspen device and the QVM, allowing you to test your code with a simulator before you reserve your slot with the device."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["tk_circ.measure_all()"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pyquil import get_qc\n", "from pytket.extensions.pyquil import ForestBackend"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["aspen_qc = get_qc(\"9q-square\", as_qvm=True)\n", "aspen_backend = ForestBackend(aspen_qc)\n", "tk_circ = aspen_backend.get_compiled_circuit(tk_circ)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["counts = aspen_backend.run_circuit(tk_circ, 2000).get_counts()\n", "print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Note that attempting to connect to a live quantum device (using a `QuantumComputer` constructed with `as_qvm=False`) will fail unless it is running from a QMI instance during a reservation for the named lattice."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Code portability and intro to forest"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The quantum hardware landscape is incredibly competitive and rapidly changing. Many full-stack quantum software platforms lock users into them in order to use the associated devices and simulators. This notebook demonstrates how `pytket` can free up your existing high-level code to be used on devices from other providers. We will take a state-preparation and evolution circuit generated using `qiskit`, and enable it to be run on several Rigetti backends.
\n", "
\n", "To use a real hardware device, this notebook should be run from a Rigetti QMI instance. Look [here](https://www.rigetti.com/qcs/docs/intro-to-qcs) for information on how to set this up. Otherwise, make sure you have QuilC and QVM running in server mode. You will need to have `pytket`, `pytket_pyquil`, and `pytket_qiskit` installed, which are all available from PyPI."]}, {"cell_type": "markdown", "metadata": {}, "source": ["We will start by using `qiskit` to build a random initial state over some qubits. (We remove the initial \"reset\" gates from the circuit since these are not recognized by the Forest backends, which assume an all-zero initial state.)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from qiskit import QuantumCircuit\n", "from qiskit.quantum_info.states.random import random_statevector"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["n_qubits = 3\n", "state = random_statevector((1 << n_qubits, 1)).data\n", "state_prep_circ = QuantumCircuit(n_qubits)\n", "state_prep_circ.initialize(state)\n", "state_prep_circ = state_prep_circ.decompose().decompose()\n", "state_prep_circ.data = [\n", " datum for datum in state_prep_circ.data if datum[0].name != \"reset\"\n", "]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(state_prep_circ)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can now evolve this state under an operator for a given duration."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from qiskit.opflow import PauliTrotterEvolution\n", "from qiskit.opflow.primitive_ops import PauliSumOp\n", "from qiskit.quantum_info import Pauli"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["duration = 1.2\n", "op = PauliSumOp.from_list([(\"XXI\", 0.3), (\"YYI\", 0.5), (\"ZZZ\", -0.4)])\n", "evolved_op = (duration * op).exp_i()\n", "evolution_circ = PauliTrotterEvolution(reps=1).convert(evolved_op).to_circuit()\n", "print(evolution_circ)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["for op in evolution_circ:\n", " state_prep_circ.append(op)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now that we have a circuit, `pytket` can take this and start operating on it directly. For example, we can apply some basic compilation passes to simplify it."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import qiskit_to_tk"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["tk_circ = qiskit_to_tk(state_prep_circ)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.passes import (\n", " SequencePass,\n", " CliffordSimp,\n", " DecomposeBoxes,\n", " KAKDecomposition,\n", " SynthesiseTket,\n", ")"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["DecomposeBoxes().apply(tk_circ)\n", "optimise = SequencePass([KAKDecomposition(), CliffordSimp(False), SynthesiseTket()])\n", "optimise.apply(tk_circ)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Display the optimised circuit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit.display import render_circuit_jupyter"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["render_circuit_jupyter(tk_circ)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The Backends in `pytket` abstract away the differences between different devices and simulators as much as possible, allowing painless switching between them. The `pytket_pyquil` package provides two Backends: `ForestBackend` encapsulates both running on physical devices via Rigetti QCS and simulating those devices on the QVM, and `ForestStateBackend` acts as a wrapper to the pyQuil Wavefunction Simulator.
\n", "
\n", "Both of these still have a few restrictions on the circuits that can be run. Each only supports a subset of the gate types available in `pytket`, and a real device or associated simulation will have restricted qubit connectivity. The Backend objects will contain a default compilation pass that will statisfy these constraints as much as possible, with minimal or no optimisation.
\n", "
\n", "The `ForestStateBackend` will allow us to view the full statevector (wavefunction) expected from a perfect execution of the circuit."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.pyquil import ForestStateBackend"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["state_backend = ForestStateBackend()\n", "tk_circ = state_backend.get_compiled_circuit(tk_circ)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["handle = state_backend.process_circuit(tk_circ)\n", "state = state_backend.get_result(handle).get_state()\n", "print(state)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["For users who are familiar with the Forest SDK, the association of qubits to indices of bitstrings (and consequently the ordering of statevectors) used by default in `pytket` Backends differs from that described in the [Forest docs](http://docs.rigetti.com/en/stable/wavefunction_simulator.html#multi-qubit-basis-enumeration). You can recover the ordering used by the Forest systems with `BackendResult.get_state(tk_circ, basis:pytket.BasisOrder.dlo)` (see our docs on the `BasisOrder` enum for more details)."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Connecting to real devices works very similarly. Instead of obtaining the full statevector, we are only able to measure the quantum state and sample from the resulting distribution. Beyond that, the process is pretty much the same.
\n", "
\n", "The following shows how to run the circuit on the \"9q-square\" lattice. The `as_qvm` switch on the `get_qc` method will switch between connecting to the real Aspen device and the QVM, allowing you to test your code with a simulator before you reserve your slot with the device."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["tk_circ.measure_all()"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pyquil import get_qc\n", "from pytket.extensions.pyquil import ForestBackend"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["aspen_qc = get_qc(\"9q-square\", as_qvm=True)\n", "aspen_backend = ForestBackend(aspen_qc)\n", "tk_circ = aspen_backend.get_compiled_circuit(tk_circ)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["counts = aspen_backend.run_circuit(tk_circ, 2000).get_counts()\n", "print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Note that attempting to connect to a live quantum device (using a `QuantumComputer` constructed with `as_qvm=False`) will fail unless it is running from a QMI instance during a reservation for the named lattice."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file diff --git a/examples/backends_example.ipynb b/examples/backends_example.ipynb index 1e008562..8ed5b4bc 100644 --- a/examples/backends_example.ipynb +++ b/examples/backends_example.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# TKET Backend Tutorial"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This example shows how to use `pytket` to execute quantum circuits on both simulators and real devices, and how to interpret the results. As tket is designed to be platform-agnostic, we have unified the interfaces of different providers as much as possible into the `Backend` class for maximum portability of code. The following is a selection of currently supported backends:
\n", "* ProjectQ simulator
\n", "* Aer simulators (statevector, QASM, and unitary)
\n", "* IBMQ devices
\n", "* Rigetti QCS devices
\n", "* Rigetti QVM (for device simulation or statevector)
\n", "* AQT devices
\n", "* Quantinuum devices
\n", "* Q# simulators"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In this notebook we will focus on the Aer, IBMQ and ProjectQ backends.
\n", "
\n", "To get started, we must install the core pytket package and the subpackages required to interface with the desired providers. We will also need the `QubitOperator` class from `openfermion` to construct operators for a later example. To get everything run the following in shell:
\n", "
\n", "`pip install pytket pytket-qiskit pytket-projectq openfermion`
\n", "
\n", "First, import the backends that we will be demonstrating."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import (\n", " AerStateBackend,\n", " AerBackend,\n", " AerUnitaryBackend,\n", " IBMQBackend,\n", " IBMQEmulatorBackend,\n", ")\n", "from pytket.extensions.projectq import ProjectQBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We are also going to be making a circuit to run on these backends, so import the `Circuit` class."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit, Qubit"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Below we generate a circuit which will produce a Bell state, assuming the qubits are all initialised in the |0> state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ = Circuit(2)\n", "circ.H(0)\n", "circ.CX(0, 1)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["As a sanity check, we will use the `AerStateBackend` to verify that `circ` does actually produce a Bell state.
\n", "
\n", "To submit a circuit for excution on a backend we can use `process_circuit` with appropriate arguments. If we have multiple circuits to excecute, we can use `process_circuits` (note the plural), which will attempt to batch up the circuits if possible. Both methods return a `ResultHandle` object per submitted `Circuit` which you can use with result retrieval methods to get the result type you want (as long as that result type is supported by the backend).
\n", "
\n", "Calling `get_state` will return a `numpy` array corresponding to the statevector.
\n", "
\n", "This style of usage is used consistently in the `pytket` backends."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["aer_state_b = AerStateBackend()\n", "state_handle = aer_state_b.process_circuit(circ)\n", "statevector = aer_state_b.get_result(state_handle).get_state()\n", "print(statevector)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["As we can see, the output state vector $\\lvert \\psi_{\\mathrm{circ}}\\rangle$ is $(\\lvert00\\rangle + \\lvert11\\rangle)/\\sqrt2$.
\n", "
\n", "This is a symmetric state. For non-symmetric states, we default to an ILO-BE format (increasing lexicographic order of (qu)bit ids, big-endian), but an alternative convention can be specified when retrieving results from backends. See the docs for the `BasisOrder` enum for more information."]}, {"cell_type": "markdown", "metadata": {}, "source": ["A lesser-used simulator available through Qiskit Aer is their unitary simulator. This will be somewhat more expensive to run, but returns the full unitary matrix for the provided circuit. This is useful in the design of small subcircuits that will be used multiple times within other larger circuits - statevector simulators will only test that they act correctly on the $\\lvert 0 \\rangle^{\\otimes n}$ state, which is not enough to guarantee the circuit's correctness.
\n", "
\n", "The `AerUnitaryBackend` provides a convenient access point for this simulator for use with `pytket` circuits. The unitary of the circuit can be retrieved from backends that support it using the `BackendResult.get_unitary` interface. In this example, we chose to use `Backend.run_circuit`, which is equivalent to calling `process_circuit` followed by `get_result`."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["aer_unitary_b = AerUnitaryBackend()\n", "result = aer_unitary_b.run_circuit(circ)\n", "print(result.get_unitary())"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Note that state vector and unitary simulations are also available in pytket directly. In general, we recommend you use these unless you require another Backend explicitly."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["statevector = circ.get_statevector()\n", "unitary = circ.get_unitary()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now suppose we want to measure this Bell state to get some actual results out, so let's append some `Measure` gates to the circuit. The `Circuit` class has the `measure_all` utility function which appends `Measure` gates on every qubit. All of these results will be written to the default classical register ('c'). This function will automatically add the classical bits to the circuit if they are not already there."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ.measure_all()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can get some measured counts out from the `AerBackend`, which is an interface to the Qiskit Aer QASM simulator. Suppose we would like to get 10 shots out (10 repeats of the circuit and measurement). We can seed the simulator's random-number generator in order to make the results reproducible, using an optional keyword argument to `process_circuit`."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["aer_b = AerBackend()\n", "handle = aer_b.process_circuit(circ, n_shots=10, seed=1)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["counts = aer_b.get_result(handle).get_counts()\n", "print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["What happens if we simulate some noise in our imagined device, using the Qiskit Aer noise model?"]}, {"cell_type": "markdown", "metadata": {}, "source": ["To investigate this, we will require an import from Qiskit. For more information about noise modelling using Qiskit Aer, see the [Qiskit device noise](https://qiskit.org/documentation/apidoc/aer_noise.html) documentation."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from qiskit.providers.aer.noise import NoiseModel"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["my_noise_model = NoiseModel()\n", "readout_error = 0.2\n", "for q in range(2):\n", " my_noise_model.add_readout_error(\n", " [[1 - readout_error, readout_error], [readout_error, 1 - readout_error]], [q]\n", " )"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This simple noise model gives a 20% chance that, upon measurement, a qubit that would otherwise have been measured as $0$ would instead be measured as $1$, and vice versa. Let's see what our shot table looks like with this model:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["noisy_aer_b = AerBackend(my_noise_model)\n", "noisy_handle = noisy_aer_b.process_circuit(circ, n_shots=10, seed=1, valid_check=False)\n", "noisy_counts = noisy_aer_b.get_result(noisy_handle).get_counts()\n", "print(noisy_counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We now have some spurious $01$ and $10$ measurements, which could never happen when measuring a Bell state on a noiseless device.
\n", "
\n", "The `AerBackend` class can accept any Qiskit noise model.
\n", "
\n", "All backends expose a generic `get_result` method which takes a `ResultHandle` and returns the respective result in the form of a `BackendResult` object. This object may hold measured results in the form of shots or counts, or an exact statevector from simulation. Measured results are stored as `OutcomeArray` objects, which compresses measured bit values into 8-bit integers. We can extract the bitwise values using `to_readouts`.
\n", "
\n", "Instead of an assumed ILO or DLO convention, we can use this object to request only the `Bit` measurements we want, in the order we want. Let's try reversing the bits of the noisy results."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend_result = noisy_aer_b.get_result(noisy_handle)\n", "bits = circ.bits"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["outcomes = backend_result.get_counts([bits[1], bits[0]])\n", "print(outcomes)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["`BackendResult` objects can be natively serialized to and deserialized from a dictionary. This dictionary can be immediately dumped to `json` for storing results."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.backends.backendresult import BackendResult"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["result_dict = backend_result.to_dict()\n", "print(result_dict)\n", "print(BackendResult.from_dict(result_dict).get_counts())"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The last simulator we will demonstrate is the `ProjectQBackend`. ProjectQ offers fast simulation of quantum circuits with built-in support for fast expectation values from operators. The `ProjectQBackend` exposes this functionality to take in OpenFermion `QubitOperator` instances. These are convertible to and from `QubitPauliOperator` instances in Pytket.
\n", "
\n", "Note: ProjectQ can also produce statevectors in the style of `AerStateBackend`, and similarly Aer backends can calculate expectation values directly, consult the relevant documentation to see more.
\n", "
\n", "Let's create an OpenFermion `QubitOperator` object and a new circuit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import openfermion as of"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian = 0.5 * of.QubitOperator(\"X0 X2\") + 0.3 * of.QubitOperator(\"Z0\")"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ2 = Circuit(3)\n", "circ2.Y(0)\n", "circ2.H(1)\n", "circ2.Rx(0.3, 2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We convert the OpenFermion Hamiltonian into a pytket QubitPauliOperator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["pauli_sym = {\"I\": Pauli.I, \"X\": Pauli.X, \"Y\": Pauli.Y, \"Z\": Pauli.Z}"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qps_from_openfermion(paulis):\n", " \"\"\"Convert OpenFermion tensor of Paulis to pytket QubitPauliString.\"\"\"\n", " qlist = []\n", " plist = []\n", " for q, p in paulis:\n", " qlist.append(Qubit(q))\n", " plist.append(pauli_sym[p])\n", " return QubitPauliString(qlist, plist)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qpo_from_openfermion(openf_op):\n", " \"\"\"Convert OpenFermion QubitOperator to pytket QubitPauliOperator.\"\"\"\n", " tk_op = dict()\n", " for term, coeff in openf_op.terms.items():\n", " string = qps_from_openfermion(term)\n", " tk_op[string] = coeff\n", " return QubitPauliOperator(tk_op)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian_op = qpo_from_openfermion(hamiltonian)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now we can create a `ProjectQBackend` instance and feed it our circuit and `QubitOperator`:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["projectq_b = ProjectQBackend()\n", "expectation = projectq_b.get_operator_expectation_value(circ2, hamiltonian_op)\n", "print(expectation)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The last leg of this tour includes running a pytket circuit on an actual quantum computer. To do this, you will need an IBM quantum experience account and have your credentials stored on your computer. See https://quantum-computing.ibm.com to make an account and view available devices and their specs.
\n", "
\n", "Physical devices have much stronger constraints on the form of admissible circuits than simulators. They tend to support a minimal gate set, have restricted connectivity between qubits for two-qubit gates, and can have limited support for classical control flow or conditional gates. This is where we can invoke the tket compiler passes to transform our desired circuit into one that is suitable for the backend.
\n", "
\n", "To check our code works correctly, we can use the `IBMQEmulatorBackend` to run our code exactly as if it were going to run on a real device, but just execute on a simulator (with a basic noise model adapted from the reported device properties)."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Let's create an `IBMQEmulatorBackend` for the `ibmq_manila` device and check if our circuit is valid to be run."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ibmq_b_emu = IBMQEmulatorBackend(\"ibmq_manila\")\n", "ibmq_b_emu.valid_circuit(circ)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["It looks like we need to compile this circuit to be compatible with the device. To simplify this procedure, we provide minimal compilation passes designed for each backend (the `default_compilation_pass()` method) which will guarantee compatibility with the device. These may still fail if the input circuit has too many qubits or unsupported usage of conditional gates. The default passes can have their degree of optimisation by changing an integer parameter (optimisation levels 0, 1, 2), and they can be easily composed with any of tket's other optimisation passes for better performance.
\n", "
\n", "For convenience, we also wrap up this pass into the `get_compiled_circuit` method if you just want to compile a single circuit."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["compiled_circ = ibmq_b_emu.get_compiled_circuit(circ)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Let's create a backend for running on the actual device and check our compiled circuit is valid for this backend too."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ibmq_b = IBMQBackend(\"ibmq_manila\")\n", "ibmq_b.valid_circuit(compiled_circ)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We are now good to run this circuit on the device. After submitting, we can use the handle to check on the status of the job, so that we know when results are ready to be retrieved. The `circuit_status` method works for all backends, and returns a `CircuitStatus` object. If we just run `get_result` straight away, the backend will wait for results to complete, blocking any other code from running.
\n", "
\n", "In this notebook we will use the emulated backend `ibmq_b_emu` to illustrate, but the workflow is the same as for the real backend `ibmq_b` (except that the latter will typically take much longer because of the size of the queue)."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["quantum_handle = ibmq_b_emu.process_circuit(compiled_circ, n_shots=10)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(ibmq_b_emu.circuit_status(quantum_handle))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["quantum_counts = ibmq_b_emu.get_result(quantum_handle).get_counts()\n", "print(quantum_counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These are from an actual device, so it's impossible to perfectly predict what the results will be. However, because of the problem of noise, it would be unsurprising to find a few $01$ or $10$ results in the table. The circuit is very short, so it should be fairly close to the ideal result.
\n", "
\n", "The devices available through the IBM Q Experience serve jobs one at a time from their respective queues, so a large amount of experiment time can be taken up by waiting for your jobs to reach the front of the queue. `pytket` allows circuits to be submitted to any backend in a single batch using the `process_circuits` method. For the `IBMQBackend`, this will collate the circuits into as few jobs as possible which will all be sent off into the queue for the device. The method returns a `ResultHandle` per submitted circuit, in the order of submission."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circuits = []\n", "for i in range(5):\n", " c = Circuit(2)\n", " c.Rx(0.2 * i, 0).CX(0, 1)\n", " c.measure_all()\n", " circuits.append(ibmq_b_emu.get_compiled_circuit(c))\n", "handles = ibmq_b_emu.process_circuits(circuits, n_shots=100)\n", "print(handles)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can now retrieve the results and process them. As we measured each circuit in the $Z$-basis, we can obtain the expectation value for the $ZZ$ operator immediately from these measurement results. We can calculate this using the `expectation_from_counts` utility method in `pytket`."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils import expectation_from_counts"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["for handle in handles:\n", " counts = ibmq_b_emu.get_result(handle).get_counts()\n", " exp_val = expectation_from_counts(counts)\n", " print(exp_val)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["A `ResultHandle` can be easily stored in its string representaton and later reconstructed using the `from_str` method. For example, we could do something like this:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.backends import ResultHandle"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(2).Rx(0.5, 0).CX(0, 1).measure_all()\n", "c = ibmq_b_emu.get_compiled_circuit(c)\n", "handle = ibmq_b_emu.process_circuit(c, n_shots=10)\n", "handlestring = str(handle)\n", "print(handlestring)\n", "# ... later ...\n", "oldhandle = ResultHandle.from_str(handlestring)\n", "print(ibmq_b_emu.get_result(oldhandle).get_counts())"]}, {"cell_type": "markdown", "metadata": {}, "source": ["For backends which support persistent handles (e.g. `IBMQBackend`, `QuantinuumBackend`, `BraketBackend` and `AQTBackend`) you can even stop your python session and use your result handles in a separate script to retrive results when they are ready, by storing the handle strings. For experiments with long queue times, this enables separate job submission and retrieval. Use `Backend.persistent_handles` to check whether a backend supports this feature.
\n", "
\n", "All backends will also cache all results obtained in the current python session, so you can use the `ResultHandle` to retrieve the results many times if you need to reuse the results. Over a long experiment, this can consume a large amount of RAM, so we recommend removing results from the cache when you are done with them. A simple way to achieve this is by calling `Backend.empty_cache` (e.g. at the end of each loop of a variational algorithm), or removing individual results with `Backend.pop_result`."]}, {"cell_type": "markdown", "metadata": {}, "source": ["The backends in `pytket` are designed to be as similar to one another as possible. The example above using physical devices can be run entirely on a simulator by swapping out the `IBMQBackend` constructor for any other backend supporting shot outputs (e.g. `AerBackend`, `ProjectQBackend`, `ForestBackend`), or passing it the name of a different device. Furthermore, using pytket it is simple to convert between handling shot tables, counts maps and statevectors.
\n", "
\n", "For more information on backends and other `pytket` features, read our [documentation](https://cqcl.github.io/pytket) or see the other examples on our [GitHub repo](https://github.com/CQCL/tket/pytket/api)."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# TKET backend tutorial"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This example shows how to use `pytket` to execute quantum circuits on both simulators and real devices, and how to interpret the results. As tket is designed to be platform-agnostic, we have unified the interfaces of different providers as much as possible into the `Backend` class for maximum portability of code. The following is a selection of currently supported backends:
\n", "* ProjectQ simulator
\n", "* Aer simulators (statevector, QASM, and unitary)
\n", "* IBMQ devices
\n", "* Rigetti QCS devices
\n", "* Rigetti QVM (for device simulation or statevector)
\n", "* AQT devices
\n", "* Quantinuum devices
\n", "* Q# simulators"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In this notebook we will focus on the Aer, IBMQ and ProjectQ backends.
\n", "
\n", "To get started, we must install the core pytket package and the subpackages required to interface with the desired providers. We will also need the `QubitOperator` class from `openfermion` to construct operators for a later example. To get everything run the following in shell:
\n", "
\n", "`pip install pytket pytket-qiskit pytket-projectq openfermion`
\n", "
\n", "First, import the backends that we will be demonstrating."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import (\n", " AerStateBackend,\n", " AerBackend,\n", " AerUnitaryBackend,\n", " IBMQBackend,\n", " IBMQEmulatorBackend,\n", ")\n", "from pytket.extensions.projectq import ProjectQBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We are also going to be making a circuit to run on these backends, so import the `Circuit` class."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit, Qubit"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Below we generate a circuit which will produce a Bell state, assuming the qubits are all initialised in the |0> state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ = Circuit(2)\n", "circ.H(0)\n", "circ.CX(0, 1)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["As a sanity check, we will use the `AerStateBackend` to verify that `circ` does actually produce a Bell state.
\n", "
\n", "To submit a circuit for excution on a backend we can use `process_circuit` with appropriate arguments. If we have multiple circuits to excecute, we can use `process_circuits` (note the plural), which will attempt to batch up the circuits if possible. Both methods return a `ResultHandle` object per submitted `Circuit` which you can use with result retrieval methods to get the result type you want (as long as that result type is supported by the backend).
\n", "
\n", "Calling `get_state` will return a `numpy` array corresponding to the statevector.
\n", "
\n", "This style of usage is used consistently in the `pytket` backends."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["aer_state_b = AerStateBackend()\n", "state_handle = aer_state_b.process_circuit(circ)\n", "statevector = aer_state_b.get_result(state_handle).get_state()\n", "print(statevector)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["As we can see, the output state vector $\\lvert \\psi_{\\mathrm{circ}}\\rangle$ is $(\\lvert00\\rangle + \\lvert11\\rangle)/\\sqrt2$.
\n", "
\n", "This is a symmetric state. For non-symmetric states, we default to an ILO-BE format (increasing lexicographic order of (qu)bit ids, big-endian), but an alternative convention can be specified when retrieving results from backends. See the docs for the `BasisOrder` enum for more information."]}, {"cell_type": "markdown", "metadata": {}, "source": ["A lesser-used simulator available through Qiskit Aer is their unitary simulator. This will be somewhat more expensive to run, but returns the full unitary matrix for the provided circuit. This is useful in the design of small subcircuits that will be used multiple times within other larger circuits - statevector simulators will only test that they act correctly on the $\\lvert 0 \\rangle^{\\otimes n}$ state, which is not enough to guarantee the circuit's correctness.
\n", "
\n", "The `AerUnitaryBackend` provides a convenient access point for this simulator for use with `pytket` circuits. The unitary of the circuit can be retrieved from backends that support it using the `BackendResult.get_unitary` interface. In this example, we chose to use `Backend.run_circuit`, which is equivalent to calling `process_circuit` followed by `get_result`."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["aer_unitary_b = AerUnitaryBackend()\n", "result = aer_unitary_b.run_circuit(circ)\n", "print(result.get_unitary())"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Note that state vector and unitary simulations are also available in pytket directly. In general, we recommend you use these unless you require another Backend explicitly."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["statevector = circ.get_statevector()\n", "unitary = circ.get_unitary()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now suppose we want to measure this Bell state to get some actual results out, so let's append some `Measure` gates to the circuit. The `Circuit` class has the `measure_all` utility function which appends `Measure` gates on every qubit. All of these results will be written to the default classical register ('c'). This function will automatically add the classical bits to the circuit if they are not already there."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ.measure_all()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can get some measured counts out from the `AerBackend`, which is an interface to the Qiskit Aer QASM simulator. Suppose we would like to get 10 shots out (10 repeats of the circuit and measurement). We can seed the simulator's random-number generator in order to make the results reproducible, using an optional keyword argument to `process_circuit`."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["aer_b = AerBackend()\n", "handle = aer_b.process_circuit(circ, n_shots=10, seed=1)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["counts = aer_b.get_result(handle).get_counts()\n", "print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["What happens if we simulate some noise in our imagined device, using the Qiskit Aer noise model?"]}, {"cell_type": "markdown", "metadata": {}, "source": ["To investigate this, we will require an import from Qiskit. For more information about noise modelling using Qiskit Aer, see the [Qiskit device noise](https://qiskit.org/documentation/apidoc/aer_noise.html) documentation."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from qiskit.providers.aer.noise import NoiseModel"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["my_noise_model = NoiseModel()\n", "readout_error = 0.2\n", "for q in range(2):\n", " my_noise_model.add_readout_error(\n", " [[1 - readout_error, readout_error], [readout_error, 1 - readout_error]], [q]\n", " )"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This simple noise model gives a 20% chance that, upon measurement, a qubit that would otherwise have been measured as $0$ would instead be measured as $1$, and vice versa. Let's see what our shot table looks like with this model:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["noisy_aer_b = AerBackend(my_noise_model)\n", "noisy_handle = noisy_aer_b.process_circuit(circ, n_shots=10, seed=1, valid_check=False)\n", "noisy_counts = noisy_aer_b.get_result(noisy_handle).get_counts()\n", "print(noisy_counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We now have some spurious $01$ and $10$ measurements, which could never happen when measuring a Bell state on a noiseless device.
\n", "
\n", "The `AerBackend` class can accept any Qiskit noise model.
\n", "
\n", "All backends expose a generic `get_result` method which takes a `ResultHandle` and returns the respective result in the form of a `BackendResult` object. This object may hold measured results in the form of shots or counts, or an exact statevector from simulation. Measured results are stored as `OutcomeArray` objects, which compresses measured bit values into 8-bit integers. We can extract the bitwise values using `to_readouts`.
\n", "
\n", "Instead of an assumed ILO or DLO convention, we can use this object to request only the `Bit` measurements we want, in the order we want. Let's try reversing the bits of the noisy results."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend_result = noisy_aer_b.get_result(noisy_handle)\n", "bits = circ.bits"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["outcomes = backend_result.get_counts([bits[1], bits[0]])\n", "print(outcomes)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["`BackendResult` objects can be natively serialized to and deserialized from a dictionary. This dictionary can be immediately dumped to `json` for storing results."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.backends.backendresult import BackendResult"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["result_dict = backend_result.to_dict()\n", "print(result_dict)\n", "print(BackendResult.from_dict(result_dict).get_counts())"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The last simulator we will demonstrate is the `ProjectQBackend`. ProjectQ offers fast simulation of quantum circuits with built-in support for fast expectation values from operators. The `ProjectQBackend` exposes this functionality to take in OpenFermion `QubitOperator` instances. These are convertible to and from `QubitPauliOperator` instances in Pytket.
\n", "
\n", "Note: ProjectQ can also produce statevectors in the style of `AerStateBackend`, and similarly Aer backends can calculate expectation values directly, consult the relevant documentation to see more.
\n", "
\n", "Let's create an OpenFermion `QubitOperator` object and a new circuit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import openfermion as of"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian = 0.5 * of.QubitOperator(\"X0 X2\") + 0.3 * of.QubitOperator(\"Z0\")"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ2 = Circuit(3)\n", "circ2.Y(0)\n", "circ2.H(1)\n", "circ2.Rx(0.3, 2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We convert the OpenFermion Hamiltonian into a pytket QubitPauliOperator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["pauli_sym = {\"I\": Pauli.I, \"X\": Pauli.X, \"Y\": Pauli.Y, \"Z\": Pauli.Z}"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qps_from_openfermion(paulis):\n", " \"\"\"Convert OpenFermion tensor of Paulis to pytket QubitPauliString.\"\"\"\n", " qlist = []\n", " plist = []\n", " for q, p in paulis:\n", " qlist.append(Qubit(q))\n", " plist.append(pauli_sym[p])\n", " return QubitPauliString(qlist, plist)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qpo_from_openfermion(openf_op):\n", " \"\"\"Convert OpenFermion QubitOperator to pytket QubitPauliOperator.\"\"\"\n", " tk_op = dict()\n", " for term, coeff in openf_op.terms.items():\n", " string = qps_from_openfermion(term)\n", " tk_op[string] = coeff\n", " return QubitPauliOperator(tk_op)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian_op = qpo_from_openfermion(hamiltonian)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now we can create a `ProjectQBackend` instance and feed it our circuit and `QubitOperator`:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["projectq_b = ProjectQBackend()\n", "expectation = projectq_b.get_operator_expectation_value(circ2, hamiltonian_op)\n", "print(expectation)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The last leg of this tour includes running a pytket circuit on an actual quantum computer. To do this, you will need an IBM quantum experience account and have your credentials stored on your computer. See https://quantum-computing.ibm.com to make an account and view available devices and their specs.
\n", "
\n", "Physical devices have much stronger constraints on the form of admissible circuits than simulators. They tend to support a minimal gate set, have restricted connectivity between qubits for two-qubit gates, and can have limited support for classical control flow or conditional gates. This is where we can invoke the tket compiler passes to transform our desired circuit into one that is suitable for the backend.
\n", "
\n", "To check our code works correctly, we can use the `IBMQEmulatorBackend` to run our code exactly as if it were going to run on a real device, but just execute on a simulator (with a basic noise model adapted from the reported device properties)."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Let's create an `IBMQEmulatorBackend` for the `ibmq_manila` device and check if our circuit is valid to be run."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ibmq_b_emu = IBMQEmulatorBackend(\"ibmq_manila\")\n", "ibmq_b_emu.valid_circuit(circ)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["It looks like we need to compile this circuit to be compatible with the device. To simplify this procedure, we provide minimal compilation passes designed for each backend (the `default_compilation_pass()` method) which will guarantee compatibility with the device. These may still fail if the input circuit has too many qubits or unsupported usage of conditional gates. The default passes can have their degree of optimisation by changing an integer parameter (optimisation levels 0, 1, 2), and they can be easily composed with any of tket's other optimisation passes for better performance.
\n", "
\n", "For convenience, we also wrap up this pass into the `get_compiled_circuit` method if you just want to compile a single circuit."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["compiled_circ = ibmq_b_emu.get_compiled_circuit(circ)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Let's create a backend for running on the actual device and check our compiled circuit is valid for this backend too."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ibmq_b = IBMQBackend(\"ibmq_manila\")\n", "ibmq_b.valid_circuit(compiled_circ)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We are now good to run this circuit on the device. After submitting, we can use the handle to check on the status of the job, so that we know when results are ready to be retrieved. The `circuit_status` method works for all backends, and returns a `CircuitStatus` object. If we just run `get_result` straight away, the backend will wait for results to complete, blocking any other code from running.
\n", "
\n", "In this notebook we will use the emulated backend `ibmq_b_emu` to illustrate, but the workflow is the same as for the real backend `ibmq_b` (except that the latter will typically take much longer because of the size of the queue)."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["quantum_handle = ibmq_b_emu.process_circuit(compiled_circ, n_shots=10)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(ibmq_b_emu.circuit_status(quantum_handle))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["quantum_counts = ibmq_b_emu.get_result(quantum_handle).get_counts()\n", "print(quantum_counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These are from an actual device, so it's impossible to perfectly predict what the results will be. However, because of the problem of noise, it would be unsurprising to find a few $01$ or $10$ results in the table. The circuit is very short, so it should be fairly close to the ideal result.
\n", "
\n", "The devices available through the IBM Q Experience serve jobs one at a time from their respective queues, so a large amount of experiment time can be taken up by waiting for your jobs to reach the front of the queue. `pytket` allows circuits to be submitted to any backend in a single batch using the `process_circuits` method. For the `IBMQBackend`, this will collate the circuits into as few jobs as possible which will all be sent off into the queue for the device. The method returns a `ResultHandle` per submitted circuit, in the order of submission."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circuits = []\n", "for i in range(5):\n", " c = Circuit(2)\n", " c.Rx(0.2 * i, 0).CX(0, 1)\n", " c.measure_all()\n", " circuits.append(ibmq_b_emu.get_compiled_circuit(c))\n", "handles = ibmq_b_emu.process_circuits(circuits, n_shots=100)\n", "print(handles)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can now retrieve the results and process them. As we measured each circuit in the $Z$-basis, we can obtain the expectation value for the $ZZ$ operator immediately from these measurement results. We can calculate this using the `expectation_from_counts` utility method in `pytket`."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils import expectation_from_counts"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["for handle in handles:\n", " counts = ibmq_b_emu.get_result(handle).get_counts()\n", " exp_val = expectation_from_counts(counts)\n", " print(exp_val)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["A `ResultHandle` can be easily stored in its string representaton and later reconstructed using the `from_str` method. For example, we could do something like this:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.backends import ResultHandle"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(2).Rx(0.5, 0).CX(0, 1).measure_all()\n", "c = ibmq_b_emu.get_compiled_circuit(c)\n", "handle = ibmq_b_emu.process_circuit(c, n_shots=10)\n", "handlestring = str(handle)\n", "print(handlestring)\n", "# ... later ...\n", "oldhandle = ResultHandle.from_str(handlestring)\n", "print(ibmq_b_emu.get_result(oldhandle).get_counts())"]}, {"cell_type": "markdown", "metadata": {}, "source": ["For backends which support persistent handles (e.g. `IBMQBackend`, `QuantinuumBackend`, `BraketBackend` and `AQTBackend`) you can even stop your python session and use your result handles in a separate script to retrive results when they are ready, by storing the handle strings. For experiments with long queue times, this enables separate job submission and retrieval. Use `Backend.persistent_handles` to check whether a backend supports this feature.
\n", "
\n", "All backends will also cache all results obtained in the current python session, so you can use the `ResultHandle` to retrieve the results many times if you need to reuse the results. Over a long experiment, this can consume a large amount of RAM, so we recommend removing results from the cache when you are done with them. A simple way to achieve this is by calling `Backend.empty_cache` (e.g. at the end of each loop of a variational algorithm), or removing individual results with `Backend.pop_result`."]}, {"cell_type": "markdown", "metadata": {}, "source": ["The backends in `pytket` are designed to be as similar to one another as possible. The example above using physical devices can be run entirely on a simulator by swapping out the `IBMQBackend` constructor for any other backend supporting shot outputs (e.g. `AerBackend`, `ProjectQBackend`, `ForestBackend`), or passing it the name of a different device. Furthermore, using pytket it is simple to convert between handling shot tables, counts maps and statevectors.
\n", "
\n", "For more information on backends and other `pytket` features, read our [documentation](https://cqcl.github.io/pytket) or see the other examples on our [GitHub repo](https://github.com/CQCL/tket/pytket/api)."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file diff --git a/examples/conditional_gate_example.ipynb b/examples/conditional_gate_example.ipynb index f2ecc9da..a481875c 100644 --- a/examples/conditional_gate_example.ipynb +++ b/examples/conditional_gate_example.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Conditional Gates"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Whilst any quantum process can be created by performing \"pure\" operations delaying all measurements to the end, this is not always practical and can greatly increase the resource requirements. It is much more convenient to alternate quantum gates and measurements, especially if we can use the measurement results to determine which gates to apply (we refer to this more generic circuit model as \"mixed\" circuits, against the usual \"pure\" circuits). This is especially crucial for error correcting codes, where the correction gates are applied only if an error is detected.
\n", "
\n", "Measurements on many NISQ devices are often slow and it is hard to maintain other qubits in a quantum state during the measurement operation. Hence they may only support a single round of measurements at the end of the circuit, removing the need for conditional gate support. However, the ability to work with mid-circuit measurement and conditional gates is a feature in high demand for the future, and tket is ready for it.
\n", "
\n", "Not every circuit language specification supports conditional gates in the same way. The most popular circuit model at the moment is that provided by the OpenQASM language. This permits a very restricted model of classical logic, where we can apply a gate conditionally on the exact value of a classical register. There is no facility in the current spec for Boolean logic or classical operations to apply any function to the value prior to the equality check.
\n", "
\n", "For example, quantum teleportation can be performed by the following QASM:
\n", "`OPENQASM 2.0;`
\n", "`include \"qelib1.inc\";`
\n", "`qreg a[2];`
\n", "`qreg b[1];`
\n", "`creg c[2];`
\n", "`// Bell state between Alice and Bob`
\n", "`h a[1];`
\n", "`cx a[1],b[0];`
\n", "`// Bell measurement of Alice's qubits`
\n", "`cx a[0],a[1];`
\n", "`h a[0];`
\n", "`measure a[0] -> c[0];`
\n", "`measure a[1] -> c[1];`
\n", "`// Correction of Bob's qubit`
\n", "`if(c==1) z b[0];`
\n", "`if(c==3) z b[0];`
\n", "`if(c==2) x b[0];`
\n", "`if(c==3) x b[0];`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["tket supports a slightly more general form of conditional gates, where the gate is applied conditionally on the exact value of any list of bits. When adding a gate to a `Circuit` object, pass in the kwargs `condition_bits` and `condition_value` and the gate will only be applied if the state of the bits yields the binary representation of the value."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.circuit.display import render_circuit_jupyter"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ = Circuit()\n", "alice = circ.add_q_register(\"a\", 2)\n", "bob = circ.add_q_register(\"b\", 1)\n", "cr = circ.add_c_register(\"c\", 2)\n", "# Bell state between Alice and Bob:\n", "circ.H(alice[1])\n", "circ.CX(alice[1], bob[0])\n", "# Bell measurement of Alice's qubits:\n", "circ.CX(alice[0], alice[1])\n", "circ.H(alice[0])\n", "circ.Measure(alice[0], cr[0])\n", "circ.Measure(alice[1], cr[1])\n", "# Correction of Bob's qubit:\n", "circ.Z(bob[0], condition_bits=[cr[0]], condition_value=1)\n", "circ.X(bob[0], condition_bits=[cr[1]], condition_value=1)\n", "render_circuit_jupyter(circ)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Performing individual gates conditionally is sufficient, but can get cumbersome for larger circuits. Fortunately, tket's Box structures can also be performed conditionally, enabling this to be applied to large circuits with ease.
\n", "
\n", "For the sake of example, assume our device struggles to perform $X$ gates. We can surround it by $CX$ gates onto an ancilla, so measuring the ancilla will either result in the identity or $X$ being applied to the target qubit. If we detect that the $X$ fails, we can retry."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import CircBox, Qubit, Bit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["checked_x = Circuit(2, 1)\n", "checked_x.CX(0, 1)\n", "checked_x.X(0)\n", "checked_x.CX(0, 1)\n", "checked_x.Measure(1, 0)\n", "x_box = CircBox(checked_x)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ2 = Circuit()\n", "target = Qubit(\"t\", 0)\n", "ancilla = Qubit(\"a\", 0)\n", "success = Bit(\"s\", 0)\n", "circ2.add_qubit(target)\n", "circ2.add_qubit(ancilla)\n", "circ2.add_bit(success)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Try the X gate:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ2.add_circbox(x_box, args=[target, ancilla, success])\n", "# Try again if the X failed\n", "circ2.add_circbox(\n", " x_box, args=[target, ancilla, success], condition_bits=[success], condition_value=0\n", ")\n", "render_circuit_jupyter(circ2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["tket is able to apply essential compilation passes on circuits containing conditional gates. This includes decomposing any boxes into primitive gates and rebasing to other gatesets whilst preserving the conditional data."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.passes import DecomposeBoxes, RebaseTket, SequencePass"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["comp_pass = SequencePass([DecomposeBoxes(), RebaseTket()])\n", "comp_pass.apply(circ2)\n", "render_circuit_jupyter(circ2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["A tket circuit can be converted to OpenQASM or other languages following the same classical model (e.g. Qiskit) when all conditional gates are dependent on the exact state of a single, whole classical register."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import tk_to_qiskit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qc = tk_to_qiskit(circ2)\n", "print(qc)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This allows us to test our mixed programs using the `AerBackend`."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import AerBackend"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ3 = Circuit(2, 1)\n", "circ3.Rx(0.3, 0)\n", "circ3.Measure(0, 0)\n", "# Set qubit 1 to be the opposite result and measure\n", "circ3.X(1, condition_bits=[0], condition_value=0)\n", "circ3.Measure(1, 0)\n", "backend = AerBackend()\n", "compiled_circ = backend.get_compiled_circuit(circ3)\n", "render_circuit_jupyter(compiled_circ)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["counts = backend.run_circuit(compiled_circ, 1024).get_counts()\n", "print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Try out mid-circuit measurement and conditional gate support on the `AerBackend` simulator, or ask about accessing the `QuantinuumBackend` to try on a hardware device."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Conditional gates"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Whilst any quantum process can be created by performing \"pure\" operations delaying all measurements to the end, this is not always practical and can greatly increase the resource requirements. It is much more convenient to alternate quantum gates and measurements, especially if we can use the measurement results to determine which gates to apply (we refer to this more generic circuit model as \"mixed\" circuits, against the usual \"pure\" circuits). This is especially crucial for error correcting codes, where the correction gates are applied only if an error is detected.
\n", "
\n", "Measurements on many NISQ devices are often slow and it is hard to maintain other qubits in a quantum state during the measurement operation. Hence they may only support a single round of measurements at the end of the circuit, removing the need for conditional gate support. However, the ability to work with mid-circuit measurement and conditional gates is a feature in high demand for the future, and tket is ready for it.
\n", "
\n", "Not every circuit language specification supports conditional gates in the same way. The most popular circuit model at the moment is that provided by the OpenQASM language. This permits a very restricted model of classical logic, where we can apply a gate conditionally on the exact value of a classical register. There is no facility in the current spec for Boolean logic or classical operations to apply any function to the value prior to the equality check.
\n", "
\n", "For example, quantum teleportation can be performed by the following QASM:
\n", "`OPENQASM 2.0;`
\n", "`include \"qelib1.inc\";`
\n", "`qreg a[2];`
\n", "`qreg b[1];`
\n", "`creg c[2];`
\n", "`// Bell state between Alice and Bob`
\n", "`h a[1];`
\n", "`cx a[1],b[0];`
\n", "`// Bell measurement of Alice's qubits`
\n", "`cx a[0],a[1];`
\n", "`h a[0];`
\n", "`measure a[0] -> c[0];`
\n", "`measure a[1] -> c[1];`
\n", "`// Correction of Bob's qubit`
\n", "`if(c==1) z b[0];`
\n", "`if(c==3) z b[0];`
\n", "`if(c==2) x b[0];`
\n", "`if(c==3) x b[0];`"]}, {"cell_type": "markdown", "metadata": {}, "source": ["tket supports a slightly more general form of conditional gates, where the gate is applied conditionally on the exact value of any list of bits. When adding a gate to a `Circuit` object, pass in the kwargs `condition_bits` and `condition_value` and the gate will only be applied if the state of the bits yields the binary representation of the value."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.circuit.display import render_circuit_jupyter"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ = Circuit()\n", "alice = circ.add_q_register(\"a\", 2)\n", "bob = circ.add_q_register(\"b\", 1)\n", "cr = circ.add_c_register(\"c\", 2)\n", "# Bell state between Alice and Bob:\n", "circ.H(alice[1])\n", "circ.CX(alice[1], bob[0])\n", "# Bell measurement of Alice's qubits:\n", "circ.CX(alice[0], alice[1])\n", "circ.H(alice[0])\n", "circ.Measure(alice[0], cr[0])\n", "circ.Measure(alice[1], cr[1])\n", "# Correction of Bob's qubit:\n", "circ.Z(bob[0], condition_bits=[cr[0]], condition_value=1)\n", "circ.X(bob[0], condition_bits=[cr[1]], condition_value=1)\n", "render_circuit_jupyter(circ)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Performing individual gates conditionally is sufficient, but can get cumbersome for larger circuits. Fortunately, tket's Box structures can also be performed conditionally, enabling this to be applied to large circuits with ease.
\n", "
\n", "For the sake of example, assume our device struggles to perform $X$ gates. We can surround it by $CX$ gates onto an ancilla, so measuring the ancilla will either result in the identity or $X$ being applied to the target qubit. If we detect that the $X$ fails, we can retry."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import CircBox, Qubit, Bit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["checked_x = Circuit(2, 1)\n", "checked_x.CX(0, 1)\n", "checked_x.X(0)\n", "checked_x.CX(0, 1)\n", "checked_x.Measure(1, 0)\n", "x_box = CircBox(checked_x)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ2 = Circuit()\n", "target = Qubit(\"t\", 0)\n", "ancilla = Qubit(\"a\", 0)\n", "success = Bit(\"s\", 0)\n", "circ2.add_qubit(target)\n", "circ2.add_qubit(ancilla)\n", "circ2.add_bit(success)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Try the X gate:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ2.add_circbox(x_box, args=[target, ancilla, success])\n", "# Try again if the X failed\n", "circ2.add_circbox(\n", " x_box, args=[target, ancilla, success], condition_bits=[success], condition_value=0\n", ")\n", "render_circuit_jupyter(circ2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["tket is able to apply essential compilation passes on circuits containing conditional gates. This includes decomposing any boxes into primitive gates and rebasing to other gatesets whilst preserving the conditional data."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.passes import DecomposeBoxes, RebaseTket, SequencePass"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["comp_pass = SequencePass([DecomposeBoxes(), RebaseTket()])\n", "comp_pass.apply(circ2)\n", "render_circuit_jupyter(circ2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["A tket circuit can be converted to OpenQASM or other languages following the same classical model (e.g. Qiskit) when all conditional gates are dependent on the exact state of a single, whole classical register."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import tk_to_qiskit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qc = tk_to_qiskit(circ2)\n", "print(qc)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This allows us to test our mixed programs using the `AerBackend`."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import AerBackend"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ3 = Circuit(2, 1)\n", "circ3.Rx(0.3, 0)\n", "circ3.Measure(0, 0)\n", "# Set qubit 1 to be the opposite result and measure\n", "circ3.X(1, condition_bits=[0], condition_value=0)\n", "circ3.Measure(1, 0)\n", "backend = AerBackend()\n", "compiled_circ = backend.get_compiled_circuit(circ3)\n", "render_circuit_jupyter(compiled_circ)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["counts = backend.run_circuit(compiled_circ, 1024).get_counts()\n", "print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Try out mid-circuit measurement and conditional gate support on the `AerBackend` simulator, or ask about accessing the `QuantinuumBackend` to try on a hardware device."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file diff --git a/examples/entanglement_swapping.ipynb b/examples/entanglement_swapping.ipynb index 8b1f5137..35896a50 100644 --- a/examples/entanglement_swapping.ipynb +++ b/examples/entanglement_swapping.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Iterated Entanglement Swapping using TKET"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In this tutorial, we will focus on:
\n", "- designing circuits with mid-circuit measurement and conditional gates;
\n", "- utilising noise models in supported simulators."]}, {"cell_type": "markdown", "metadata": {}, "source": ["This example assumes the reader is familiar with the Qubit Teleportation and Entanglement Swapping protocols, and basic models of noise in quantum devices.
\n", "
\n", "To run this example, you will need `pytket`, `pytket-qiskit`, and `plotly` (installed via `pip`). To view the graphs, you will need an intallation of `plotly-orca`.
\n", "
\n", "Current quantum hardware fits into the NISQ (Noisy, Intermediate-Scale Quantum) regime. This noise cannot realistically be combatted using conventional error correcting codes, because of the lack of available qubits, noise levels exceeding the code thresholds, and very few devices available that can perform measurements and corrections mid-circuit. Analysis of how quantum algorithms perform under noisy conditions is a very active research area, as is finding ways to cope with it. Here, we will look at how well we can perform the Entanglement Swapping protocol with different noise levels.
\n", "
\n", "The Entanglement Swapping protocol requires two parties to share Bell pairs with a third party, who applies the Qubit Teleportation protocol to generate a Bell pair between the two parties. The Qubit Teleportation step requires us to be able to measure some qubits and make subsequent corrections to the remaining qubits. There are only a handful of simulators and devices that currently support this, with others restricted to only measuring the qubits at the end of the circuit.
\n", "
\n", "The most popular circuit model with conditional gates at the moment is that provided by the OpenQASM language. This permits a very restricted model of classical logic, where we can apply a gate conditionally on the exact value of a classical register. There is no facility in the current spec for Boolean logic or classical operations to apply any function to the value prior to the equality check. For example, Qubit Teleportation can be performed by the following QASM:
\n", "`OPENQASM 2.0;`
\n", "`include \"qelib1.inc\";`
\n", "`qreg a[2];`
\n", "`qreg b[1];`
\n", "`creg c[2];`
\n", "`// Bell state between Alice and Bob`
\n", "`h a[1];`
\n", "`cx a[1],b[0];`
\n", "`// Bell measurement of Alice's qubits`
\n", "`cx a[0],a[1];`
\n", "`h a[0];`
\n", "`measure a[0] -> c[0];`
\n", "`measure a[1] -> c[1];`
\n", "`// Correction of Bob's qubit`
\n", "`if(c==1) z b[0];`
\n", "`if(c==3) z b[0];`
\n", "`if(c==2) x b[0];`
\n", "`if(c==3) x b[0];`
\n", "
\n", "This corresponds to the following `pytket` code:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qtel = Circuit()\n", "alice = qtel.add_q_register(\"a\", 2)\n", "bob = qtel.add_q_register(\"b\", 1)\n", "data = qtel.add_c_register(\"d\", 2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Bell state between Alice and Bob:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qtel.H(alice[1])\n", "qtel.CX(alice[1], bob[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Bell measurement of Alice's qubits:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qtel.CX(alice[0], alice[1])\n", "qtel.H(alice[0])\n", "qtel.Measure(alice[0], data[0])\n", "qtel.Measure(alice[1], data[1])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Correction of Bob's qubit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qtel.X(bob[0], condition_bits=[data[0], data[1]], condition_value=2)\n", "qtel.X(bob[0], condition_bits=[data[0], data[1]], condition_value=3)\n", "qtel.Z(bob[0], condition_bits=[data[0], data[1]], condition_value=1)\n", "qtel.Z(bob[0], condition_bits=[data[0], data[1]], condition_value=3)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["So to demonstrate the Entanglement Swapping protocol, we just need to run this on one side of a Bell pair."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["es = Circuit()\n", "ava = es.add_q_register(\"a\", 1)\n", "bella = es.add_q_register(\"b\", 2)\n", "charlie = es.add_q_register(\"c\", 1)\n", "data = es.add_c_register(\"d\", 2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Bell state between Ava and Bella:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["es.H(ava[0])\n", "es.CX(ava[0], bella[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Teleport `bella[0]` to `charlie[0]`:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["tel_to_c = qtel.copy()\n", "tel_to_c.rename_units({alice[0]: bella[0], alice[1]: bella[1], bob[0]: charlie[0]})\n", "es.append(tel_to_c)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(es.get_commands())"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Let's start by running a noiseless simulation of this to verify that what we get looks like a Bell pair."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import AerBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Connect to a simulator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Make a ZZ measurement of the Bell pair:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["bell_test = es.copy()\n", "bell_test.Measure(ava[0], data[0])\n", "bell_test.Measure(charlie[0], data[1])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run the experiment:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["bell_test = backend.get_compiled_circuit(bell_test)\n", "from pytket.circuit.display import render_circuit_jupyter"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["render_circuit_jupyter(bell_test)\n", "handle = backend.process_circuit(bell_test, n_shots=2000)\n", "counts = backend.get_result(handle).get_counts()"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is good, we have got roughly 50/50 measurement results of 00 and 11 under the ZZ operator. But there are many other states beyond the Bell state that also generate this distribution, so to gain more confidence in our claim about the state we should make more measurements that also characterise it, i.e. perform state tomography.
\n", "
\n", "Here, we will demonstrate a naive approach to tomography that makes 3^n measurement circuits for an n-qubit state. More elaborate methods also exist."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.utils import append_pauli_measurement, probs_from_counts\n", "from itertools import product\n", "from scipy.linalg import lstsq, eigh\n", "import numpy as np"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def gen_tomography_circuits(state, qubits, bits):\n", " # Yields {X, Y, Z}^n measurements in lexicographical order\n", " # Only measures qubits, storing the result in bits\n", " # (since we don't care about the ancilla qubits)\n", " assert len(qubits) == len(bits)\n", " for paulis in product([Pauli.X, Pauli.Y, Pauli.Z], repeat=len(qubits)):\n", " circ = state.copy()\n", " for qb, b, p in zip(qubits, bits, paulis):\n", " if p == Pauli.X:\n", " circ.H(qb)\n", " elif p == Pauli.Y:\n", " circ.V(qb)\n", " circ.Measure(qb, b)\n", " yield circ"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def run_tomography_circuits(state, qubits, bits, backend):\n", " circs = list(gen_tomography_circuits(state, qubits, bits))\n", " # Compile and run each circuit\n", " circs = backend.get_compiled_circuits(circs)\n", " handles = backend.process_circuits(circs, n_shots=2000)\n", " # Get the observed measurement probabilities\n", " probs_list = []\n", " for result in backend.get_results(handles):\n", " counts = result.get_counts()\n", " probs = probs_from_counts(counts)\n", " probs_list.append(probs)\n", " return probs_list"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def fit_tomography_outcomes(probs_list, n_qbs):\n", " # Define the density matrices for the basis states\n", " basis = dict()\n", " basis[(Pauli.X, 0)] = np.asarray([[0.5, 0.5], [0.5, 0.5]])\n", " basis[(Pauli.X, 1)] = np.asarray([[0.5, -0.5], [-0.5, 0.5]])\n", " basis[(Pauli.Y, 0)] = np.asarray([[0.5, -0.5j], [0.5j, 0.5]])\n", " basis[(Pauli.Y, 1)] = np.asarray([[0.5, 0.5j], [-0.5j, 0.5]])\n", " basis[(Pauli.Z, 0)] = np.asarray([[1, 0], [0, 0]])\n", " basis[(Pauli.Z, 1)] = np.asarray([[0, 0], [0, 1]])\n", " dim = 2**n_qbs\n", " # Define vector all_probs as a concatenation of probability vectors for each measurement (2**n x 3**n, 1)\n", " # Define matrix all_ops mapping a (vectorised) density matrix to a vector of probabilities for each measurement\n", " # (2**n x 3**n, 2**n x 2**n)\n", " all_probs = []\n", " all_ops = []\n", " for paulis, probs in zip(\n", " product([Pauli.X, Pauli.Y, Pauli.Z], repeat=n_qbs), probs_list\n", " ):\n", " prob_vec = []\n", " meas_ops = []\n", " for outcome in product([0, 1], repeat=n_qbs):\n", " prob_vec.append(probs.get(outcome, 0))\n", " op = np.eye(1, dtype=complex)\n", " for p, o in zip(paulis, outcome):\n", " op = np.kron(op, basis[(p, o)])\n", " meas_ops.append(op.reshape(1, dim * dim).conj())\n", " all_probs.append(np.vstack(prob_vec))\n", " all_ops.append(np.vstack(meas_ops))\n", " # Solve for density matrix by minimising || all_ops * dm - all_probs ||\n", " dm, _, _, _ = lstsq(np.vstack(all_ops), np.vstack(all_probs))\n", " dm = dm.reshape(dim, dim)\n", " # Make density matrix positive semi-definite\n", " v, w = eigh(dm)\n", " for i in range(dim):\n", " if v[i] < 0:\n", " for j in range(i + 1, dim):\n", " v[j] += v[i] / (dim - (i + 1))\n", " v[i] = 0\n", " dm = np.zeros([dim, dim], dtype=complex)\n", " for j in range(dim):\n", " dm += v[j] * np.outer(w[:, j], np.conj(w[:, j]))\n", " # Normalise trace of density matrix\n", " dm /= np.trace(dm)\n", " return dm"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["probs_list = run_tomography_circuits(\n", " es, [ava[0], charlie[0]], [data[0], data[1]], backend\n", ")\n", "dm = fit_tomography_outcomes(probs_list, 2)\n", "print(dm.round(3))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is very close to the true density matrix for a pure Bell state. We can attribute the error here to the sampling error since we only take 2000 samples of each measurement circuit.
\n", "
\n", "To quantify exactly how similar it is to the correct density matrix, we can calculate the fidelity."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from scipy.linalg import sqrtm"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def fidelity(dm0, dm1):\n", " # Calculate the fidelity between two density matrices\n", " sq0 = sqrtm(dm0)\n", " sq1 = sqrtm(dm1)\n", " return np.linalg.norm(sq0.dot(sq1)) ** 2"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["bell_state = np.asarray(\n", " [\n", " [0.5, 0, 0, 0.5],\n", " [0, 0, 0, 0],\n", " [0, 0, 0, 0],\n", " [0.5, 0, 0, 0.5],\n", " ]\n", ")\n", "print(fidelity(dm, bell_state))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This high fidelity is unsurprising since we have a completely noiseless simulation. So the next step is to add some noise to the simulation and observe how the overall fidelity is affected. The `AerBackend` wraps around the Qiskit Aer simulator and can pass on any `qiskit.providers.aer.noise.NoiseModel` to the simulator. Let's start by adding some uniform depolarising noise to each CX gate and some uniform measurement error."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from qiskit.providers.aer.noise import NoiseModel, depolarizing_error, ReadoutError"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def make_noise_model(dep_err_rate, ro_err_rate, qubits):\n", " # Define a noise model that applies uniformly to the given qubits\n", " model = NoiseModel()\n", " dep_err = depolarizing_error(dep_err_rate, 2)\n", " ro_err = ReadoutError(\n", " [[1 - ro_err_rate, ro_err_rate], [ro_err_rate, 1 - ro_err_rate]]\n", " )\n", " # Add depolarising error to CX gates between any qubits (implying full connectivity)\n", " for i, j in product(qubits, repeat=2):\n", " if i != j:\n", " model.add_quantum_error(dep_err, [\"cx\"], [i, j])\n", " # Add readout error for each qubit\n", " for i in qubits:\n", " model.add_readout_error(ro_err, qubits=[i])\n", " return model"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_model = make_noise_model(0.03, 0.05, range(4))\n", "backend = AerBackend(noise_model=test_model)\n", "probs_list = run_tomography_circuits(\n", " es, [ava[0], charlie[0]], [data[0], data[1]], backend\n", ")\n", "dm = fit_tomography_outcomes(probs_list, 2)\n", "print(dm.round(3))\n", "print(fidelity(dm, bell_state))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Despite the very small circuit and the relatively small error rates, the fidelity of the final state has reduced considerably.
\n", "
\n", "As far as circuits go, the entanglement swapping protocol is little more than a toy example and is nothing close to the scale of circuits for most interesting quantum computational problems. However, it is possible to iterate the protocol many times to build up a larger computation, allowing us to see the impact of the noise at different scales."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import OpType\n", "from plotly.graph_objects import Scatter, Figure"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def iterated_entanglement_swap(n_iter):\n", " # Iterate the entanglement swapping protocol n_iter times\n", " it_es = Circuit()\n", " ava = it_es.add_q_register(\"a\", 1)\n", " bella = it_es.add_q_register(\"b\", 2)\n", " charlie = it_es.add_q_register(\"c\", 1)\n", " data = it_es.add_c_register(\"d\", 2)\n\n", " # Start with an initial Bell state\n", " it_es.H(ava[0])\n", " it_es.CX(ava[0], bella[0])\n", " for i in range(n_iter):\n", " if i % 2 == 0:\n", " # Teleport bella[0] to charlie[0] to give a Bell pair between ava[0] and charlier[0]\n", " tel_to_c = qtel.copy()\n", " tel_to_c.rename_units(\n", " {alice[0]: bella[0], alice[1]: bella[1], bob[0]: charlie[0]}\n", " )\n", " it_es.append(tel_to_c)\n", " it_es.add_gate(OpType.Reset, [bella[0]])\n", " it_es.add_gate(OpType.Reset, [bella[1]])\n", " else:\n", " # Teleport charlie[0] to bella[0] to give a Bell pair between ava[0] and bella[0]\n", " tel_to_b = qtel.copy()\n", " tel_to_b.rename_units(\n", " {alice[0]: charlie[0], alice[1]: bella[1], bob[0]: bella[0]}\n", " )\n", " it_es.append(tel_to_b)\n", " it_es.add_gate(OpType.Reset, [bella[1]])\n", " it_es.add_gate(OpType.Reset, [charlie[0]])\n", " # Return the circuit and the qubits expected to share a Bell pair\n", " if n_iter % 2 == 0:\n", " return it_es, [ava[0], bella[0]]\n", " else:\n", " return it_es, [ava[0], charlie[0]]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def iterated_noisy_experiment(dep_err_rate, ro_err_rate, max_iter):\n", " # Set up the noisy simulator with the given error rates\n", " test_model = make_noise_model(dep_err_rate, ro_err_rate, range(4))\n", " backend = AerBackend(noise_model=test_model)\n", " # Estimate the fidelity after n iterations, from 0 to max_iter (inclusive)\n", " fid_list = []\n", " for i in range(max_iter + 1):\n", " it_es, qubits = iterated_entanglement_swap(i)\n", " probs_list = run_tomography_circuits(it_es, qubits, [data[0], data[1]], backend)\n", " dm = fit_tomography_outcomes(probs_list, 2)\n", " fid = fidelity(dm, bell_state)\n", " fid_list.append(fid)\n", " return fid_list"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["fig = Figure()\n", "fig.update_layout(\n", " title=\"Iterated Entanglement Swapping under Noise (dep_err = 0.03)\",\n", " xaxis_title=\"Iterations\",\n", " xaxis=dict(range=[0, 10]),\n", " yaxis_title=\"Fidelity\",\n", ")\n", "iter_range = np.arange(11)\n", "for i in range(7):\n", " fids = iterated_noisy_experiment(0.03, 0.025 * i, 10)\n", " plot_data = Scatter(\n", " x=iter_range, y=fids, name=\"ro_err=\" + str(np.round(0.025 * i, 3))\n", " )\n", " fig.add_trace(plot_data)\n", "try:\n", " fig.show(renderer=\"svg\")\n", "except ValueError as e:\n", " print(e) # requires plotly-orca"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["fig = Figure()\n", "fig.update_layout(\n", " title=\"Iterated Entanglement Swapping under Noise (ro_err = 0.05)\",\n", " xaxis_title=\"Iterations\",\n", " xaxis=dict(range=[0, 10]),\n", " yaxis_title=\"Fidelity\",\n", ")\n", "iter_range = np.arange(11)\n", "for i in range(9):\n", " fids = iterated_noisy_experiment(0.01 * i, 0.05, 10)\n", " plot_data = Scatter(\n", " x=iter_range, y=fids, name=\"dep_err=\" + str(np.round(0.01 * i, 3))\n", " )\n", " fig.add_trace(plot_data)\n", "try:\n", " fig.show(renderer=\"svg\")\n", "except ValueError as e:\n", " print(e) # requires plotly-orca"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These graphs are not very surprising, but are still important for seeing that the current error rates of typical NISQ devices become crippling for fidelities very quickly after repeated mid-circuit measurements and corrections (even with this overly-simplified model with uniform noise and no crosstalk or higher error modes). This provides good motivation for the adoption of error mitigation techniques, and for the development of new techniques that are robust to errors in mid-circuit measurements."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Exercises:
\n", "- Vary the fixed noise levels to compare how impactful the depolarising and measurement errors are.
\n", "- Add extra noise characteristics to the noise model to obtain something that more resembles a real device. Possible options include adding error during the reset operations, extending the errors to be non-local, or constructing the noise model from a device's calibration data.
\n", "- Change the circuit from iterated entanglement swapping to iterated applications of a correction circuit from a simple error-correcting code. Do you expect this to be more sensitive to depolarising errors from unitary gates or measurement errors?"]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Iterated entanglement swapping using TKET"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In this tutorial, we will focus on:
\n", "- designing circuits with mid-circuit measurement and conditional gates;
\n", "- utilising noise models in supported simulators."]}, {"cell_type": "markdown", "metadata": {}, "source": ["This example assumes the reader is familiar with the Qubit Teleportation and Entanglement Swapping protocols, and basic models of noise in quantum devices.
\n", "
\n", "To run this example, you will need `pytket`, `pytket-qiskit`, and `plotly` (installed via `pip`). To view the graphs, you will need an intallation of `plotly-orca`.
\n", "
\n", "Current quantum hardware fits into the NISQ (Noisy, Intermediate-Scale Quantum) regime. This noise cannot realistically be combatted using conventional error correcting codes, because of the lack of available qubits, noise levels exceeding the code thresholds, and very few devices available that can perform measurements and corrections mid-circuit. Analysis of how quantum algorithms perform under noisy conditions is a very active research area, as is finding ways to cope with it. Here, we will look at how well we can perform the Entanglement Swapping protocol with different noise levels.
\n", "
\n", "The Entanglement Swapping protocol requires two parties to share Bell pairs with a third party, who applies the Qubit Teleportation protocol to generate a Bell pair between the two parties. The Qubit Teleportation step requires us to be able to measure some qubits and make subsequent corrections to the remaining qubits. There are only a handful of simulators and devices that currently support this, with others restricted to only measuring the qubits at the end of the circuit.
\n", "
\n", "The most popular circuit model with conditional gates at the moment is that provided by the OpenQASM language. This permits a very restricted model of classical logic, where we can apply a gate conditionally on the exact value of a classical register. There is no facility in the current spec for Boolean logic or classical operations to apply any function to the value prior to the equality check. For example, Qubit Teleportation can be performed by the following QASM:
\n", "`OPENQASM 2.0;`
\n", "`include \"qelib1.inc\";`
\n", "`qreg a[2];`
\n", "`qreg b[1];`
\n", "`creg c[2];`
\n", "`// Bell state between Alice and Bob`
\n", "`h a[1];`
\n", "`cx a[1],b[0];`
\n", "`// Bell measurement of Alice's qubits`
\n", "`cx a[0],a[1];`
\n", "`h a[0];`
\n", "`measure a[0] -> c[0];`
\n", "`measure a[1] -> c[1];`
\n", "`// Correction of Bob's qubit`
\n", "`if(c==1) z b[0];`
\n", "`if(c==3) z b[0];`
\n", "`if(c==2) x b[0];`
\n", "`if(c==3) x b[0];`
\n", "
\n", "This corresponds to the following `pytket` code:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qtel = Circuit()\n", "alice = qtel.add_q_register(\"a\", 2)\n", "bob = qtel.add_q_register(\"b\", 1)\n", "data = qtel.add_c_register(\"d\", 2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Bell state between Alice and Bob:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qtel.H(alice[1])\n", "qtel.CX(alice[1], bob[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Bell measurement of Alice's qubits:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qtel.CX(alice[0], alice[1])\n", "qtel.H(alice[0])\n", "qtel.Measure(alice[0], data[0])\n", "qtel.Measure(alice[1], data[1])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Correction of Bob's qubit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qtel.X(bob[0], condition_bits=[data[0], data[1]], condition_value=2)\n", "qtel.X(bob[0], condition_bits=[data[0], data[1]], condition_value=3)\n", "qtel.Z(bob[0], condition_bits=[data[0], data[1]], condition_value=1)\n", "qtel.Z(bob[0], condition_bits=[data[0], data[1]], condition_value=3)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["So to demonstrate the Entanglement Swapping protocol, we just need to run this on one side of a Bell pair."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["es = Circuit()\n", "ava = es.add_q_register(\"a\", 1)\n", "bella = es.add_q_register(\"b\", 2)\n", "charlie = es.add_q_register(\"c\", 1)\n", "data = es.add_c_register(\"d\", 2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Bell state between Ava and Bella:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["es.H(ava[0])\n", "es.CX(ava[0], bella[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Teleport `bella[0]` to `charlie[0]`:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["tel_to_c = qtel.copy()\n", "tel_to_c.rename_units({alice[0]: bella[0], alice[1]: bella[1], bob[0]: charlie[0]})\n", "es.append(tel_to_c)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(es.get_commands())"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Let's start by running a noiseless simulation of this to verify that what we get looks like a Bell pair."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import AerBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Connect to a simulator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Make a ZZ measurement of the Bell pair:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["bell_test = es.copy()\n", "bell_test.Measure(ava[0], data[0])\n", "bell_test.Measure(charlie[0], data[1])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run the experiment:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["bell_test = backend.get_compiled_circuit(bell_test)\n", "from pytket.circuit.display import render_circuit_jupyter"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["render_circuit_jupyter(bell_test)\n", "handle = backend.process_circuit(bell_test, n_shots=2000)\n", "counts = backend.get_result(handle).get_counts()"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is good, we have got roughly 50/50 measurement results of 00 and 11 under the ZZ operator. But there are many other states beyond the Bell state that also generate this distribution, so to gain more confidence in our claim about the state we should make more measurements that also characterise it, i.e. perform state tomography.
\n", "
\n", "Here, we will demonstrate a naive approach to tomography that makes 3^n measurement circuits for an n-qubit state. More elaborate methods also exist."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.utils import append_pauli_measurement, probs_from_counts\n", "from itertools import product\n", "from scipy.linalg import lstsq, eigh\n", "import numpy as np"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def gen_tomography_circuits(state, qubits, bits):\n", " # Yields {X, Y, Z}^n measurements in lexicographical order\n", " # Only measures qubits, storing the result in bits\n", " # (since we don't care about the ancilla qubits)\n", " assert len(qubits) == len(bits)\n", " for paulis in product([Pauli.X, Pauli.Y, Pauli.Z], repeat=len(qubits)):\n", " circ = state.copy()\n", " for qb, b, p in zip(qubits, bits, paulis):\n", " if p == Pauli.X:\n", " circ.H(qb)\n", " elif p == Pauli.Y:\n", " circ.V(qb)\n", " circ.Measure(qb, b)\n", " yield circ"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def run_tomography_circuits(state, qubits, bits, backend):\n", " circs = list(gen_tomography_circuits(state, qubits, bits))\n", " # Compile and run each circuit\n", " circs = backend.get_compiled_circuits(circs)\n", " handles = backend.process_circuits(circs, n_shots=2000)\n", " # Get the observed measurement probabilities\n", " probs_list = []\n", " for result in backend.get_results(handles):\n", " counts = result.get_counts()\n", " probs = probs_from_counts(counts)\n", " probs_list.append(probs)\n", " return probs_list"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def fit_tomography_outcomes(probs_list, n_qbs):\n", " # Define the density matrices for the basis states\n", " basis = dict()\n", " basis[(Pauli.X, 0)] = np.asarray([[0.5, 0.5], [0.5, 0.5]])\n", " basis[(Pauli.X, 1)] = np.asarray([[0.5, -0.5], [-0.5, 0.5]])\n", " basis[(Pauli.Y, 0)] = np.asarray([[0.5, -0.5j], [0.5j, 0.5]])\n", " basis[(Pauli.Y, 1)] = np.asarray([[0.5, 0.5j], [-0.5j, 0.5]])\n", " basis[(Pauli.Z, 0)] = np.asarray([[1, 0], [0, 0]])\n", " basis[(Pauli.Z, 1)] = np.asarray([[0, 0], [0, 1]])\n", " dim = 2**n_qbs\n", " # Define vector all_probs as a concatenation of probability vectors for each measurement (2**n x 3**n, 1)\n", " # Define matrix all_ops mapping a (vectorised) density matrix to a vector of probabilities for each measurement\n", " # (2**n x 3**n, 2**n x 2**n)\n", " all_probs = []\n", " all_ops = []\n", " for paulis, probs in zip(\n", " product([Pauli.X, Pauli.Y, Pauli.Z], repeat=n_qbs), probs_list\n", " ):\n", " prob_vec = []\n", " meas_ops = []\n", " for outcome in product([0, 1], repeat=n_qbs):\n", " prob_vec.append(probs.get(outcome, 0))\n", " op = np.eye(1, dtype=complex)\n", " for p, o in zip(paulis, outcome):\n", " op = np.kron(op, basis[(p, o)])\n", " meas_ops.append(op.reshape(1, dim * dim).conj())\n", " all_probs.append(np.vstack(prob_vec))\n", " all_ops.append(np.vstack(meas_ops))\n", " # Solve for density matrix by minimising || all_ops * dm - all_probs ||\n", " dm, _, _, _ = lstsq(np.vstack(all_ops), np.vstack(all_probs))\n", " dm = dm.reshape(dim, dim)\n", " # Make density matrix positive semi-definite\n", " v, w = eigh(dm)\n", " for i in range(dim):\n", " if v[i] < 0:\n", " for j in range(i + 1, dim):\n", " v[j] += v[i] / (dim - (i + 1))\n", " v[i] = 0\n", " dm = np.zeros([dim, dim], dtype=complex)\n", " for j in range(dim):\n", " dm += v[j] * np.outer(w[:, j], np.conj(w[:, j]))\n", " # Normalise trace of density matrix\n", " dm /= np.trace(dm)\n", " return dm"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["probs_list = run_tomography_circuits(\n", " es, [ava[0], charlie[0]], [data[0], data[1]], backend\n", ")\n", "dm = fit_tomography_outcomes(probs_list, 2)\n", "print(dm.round(3))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is very close to the true density matrix for a pure Bell state. We can attribute the error here to the sampling error since we only take 2000 samples of each measurement circuit.
\n", "
\n", "To quantify exactly how similar it is to the correct density matrix, we can calculate the fidelity."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from scipy.linalg import sqrtm"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def fidelity(dm0, dm1):\n", " # Calculate the fidelity between two density matrices\n", " sq0 = sqrtm(dm0)\n", " sq1 = sqrtm(dm1)\n", " return np.linalg.norm(sq0.dot(sq1)) ** 2"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["bell_state = np.asarray(\n", " [\n", " [0.5, 0, 0, 0.5],\n", " [0, 0, 0, 0],\n", " [0, 0, 0, 0],\n", " [0.5, 0, 0, 0.5],\n", " ]\n", ")\n", "print(fidelity(dm, bell_state))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This high fidelity is unsurprising since we have a completely noiseless simulation. So the next step is to add some noise to the simulation and observe how the overall fidelity is affected. The `AerBackend` wraps around the Qiskit Aer simulator and can pass on any `qiskit.providers.aer.noise.NoiseModel` to the simulator. Let's start by adding some uniform depolarising noise to each CX gate and some uniform measurement error."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from qiskit.providers.aer.noise import NoiseModel, depolarizing_error, ReadoutError"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def make_noise_model(dep_err_rate, ro_err_rate, qubits):\n", " # Define a noise model that applies uniformly to the given qubits\n", " model = NoiseModel()\n", " dep_err = depolarizing_error(dep_err_rate, 2)\n", " ro_err = ReadoutError(\n", " [[1 - ro_err_rate, ro_err_rate], [ro_err_rate, 1 - ro_err_rate]]\n", " )\n", " # Add depolarising error to CX gates between any qubits (implying full connectivity)\n", " for i, j in product(qubits, repeat=2):\n", " if i != j:\n", " model.add_quantum_error(dep_err, [\"cx\"], [i, j])\n", " # Add readout error for each qubit\n", " for i in qubits:\n", " model.add_readout_error(ro_err, qubits=[i])\n", " return model"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_model = make_noise_model(0.03, 0.05, range(4))\n", "backend = AerBackend(noise_model=test_model)\n", "probs_list = run_tomography_circuits(\n", " es, [ava[0], charlie[0]], [data[0], data[1]], backend\n", ")\n", "dm = fit_tomography_outcomes(probs_list, 2)\n", "print(dm.round(3))\n", "print(fidelity(dm, bell_state))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Despite the very small circuit and the relatively small error rates, the fidelity of the final state has reduced considerably.
\n", "
\n", "As far as circuits go, the entanglement swapping protocol is little more than a toy example and is nothing close to the scale of circuits for most interesting quantum computational problems. However, it is possible to iterate the protocol many times to build up a larger computation, allowing us to see the impact of the noise at different scales."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import OpType\n", "from plotly.graph_objects import Scatter, Figure"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def iterated_entanglement_swap(n_iter):\n", " # Iterate the entanglement swapping protocol n_iter times\n", " it_es = Circuit()\n", " ava = it_es.add_q_register(\"a\", 1)\n", " bella = it_es.add_q_register(\"b\", 2)\n", " charlie = it_es.add_q_register(\"c\", 1)\n", " data = it_es.add_c_register(\"d\", 2)\n\n", " # Start with an initial Bell state\n", " it_es.H(ava[0])\n", " it_es.CX(ava[0], bella[0])\n", " for i in range(n_iter):\n", " if i % 2 == 0:\n", " # Teleport bella[0] to charlie[0] to give a Bell pair between ava[0] and charlier[0]\n", " tel_to_c = qtel.copy()\n", " tel_to_c.rename_units(\n", " {alice[0]: bella[0], alice[1]: bella[1], bob[0]: charlie[0]}\n", " )\n", " it_es.append(tel_to_c)\n", " it_es.add_gate(OpType.Reset, [bella[0]])\n", " it_es.add_gate(OpType.Reset, [bella[1]])\n", " else:\n", " # Teleport charlie[0] to bella[0] to give a Bell pair between ava[0] and bella[0]\n", " tel_to_b = qtel.copy()\n", " tel_to_b.rename_units(\n", " {alice[0]: charlie[0], alice[1]: bella[1], bob[0]: bella[0]}\n", " )\n", " it_es.append(tel_to_b)\n", " it_es.add_gate(OpType.Reset, [bella[1]])\n", " it_es.add_gate(OpType.Reset, [charlie[0]])\n", " # Return the circuit and the qubits expected to share a Bell pair\n", " if n_iter % 2 == 0:\n", " return it_es, [ava[0], bella[0]]\n", " else:\n", " return it_es, [ava[0], charlie[0]]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def iterated_noisy_experiment(dep_err_rate, ro_err_rate, max_iter):\n", " # Set up the noisy simulator with the given error rates\n", " test_model = make_noise_model(dep_err_rate, ro_err_rate, range(4))\n", " backend = AerBackend(noise_model=test_model)\n", " # Estimate the fidelity after n iterations, from 0 to max_iter (inclusive)\n", " fid_list = []\n", " for i in range(max_iter + 1):\n", " it_es, qubits = iterated_entanglement_swap(i)\n", " probs_list = run_tomography_circuits(it_es, qubits, [data[0], data[1]], backend)\n", " dm = fit_tomography_outcomes(probs_list, 2)\n", " fid = fidelity(dm, bell_state)\n", " fid_list.append(fid)\n", " return fid_list"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["fig = Figure()\n", "fig.update_layout(\n", " title=\"Iterated Entanglement Swapping under Noise (dep_err = 0.03)\",\n", " xaxis_title=\"Iterations\",\n", " xaxis=dict(range=[0, 10]),\n", " yaxis_title=\"Fidelity\",\n", ")\n", "iter_range = np.arange(11)\n", "for i in range(7):\n", " fids = iterated_noisy_experiment(0.03, 0.025 * i, 10)\n", " plot_data = Scatter(\n", " x=iter_range, y=fids, name=\"ro_err=\" + str(np.round(0.025 * i, 3))\n", " )\n", " fig.add_trace(plot_data)\n", "try:\n", " fig.show(renderer=\"svg\")\n", "except ValueError as e:\n", " print(e) # requires plotly-orca"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["fig = Figure()\n", "fig.update_layout(\n", " title=\"Iterated Entanglement Swapping under Noise (ro_err = 0.05)\",\n", " xaxis_title=\"Iterations\",\n", " xaxis=dict(range=[0, 10]),\n", " yaxis_title=\"Fidelity\",\n", ")\n", "iter_range = np.arange(11)\n", "for i in range(9):\n", " fids = iterated_noisy_experiment(0.01 * i, 0.05, 10)\n", " plot_data = Scatter(\n", " x=iter_range, y=fids, name=\"dep_err=\" + str(np.round(0.01 * i, 3))\n", " )\n", " fig.add_trace(plot_data)\n", "try:\n", " fig.show(renderer=\"svg\")\n", "except ValueError as e:\n", " print(e) # requires plotly-orca"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These graphs are not very surprising, but are still important for seeing that the current error rates of typical NISQ devices become crippling for fidelities very quickly after repeated mid-circuit measurements and corrections (even with this overly-simplified model with uniform noise and no crosstalk or higher error modes). This provides good motivation for the adoption of error mitigation techniques, and for the development of new techniques that are robust to errors in mid-circuit measurements."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Exercises:
\n", "- Vary the fixed noise levels to compare how impactful the depolarising and measurement errors are.
\n", "- Add extra noise characteristics to the noise model to obtain something that more resembles a real device. Possible options include adding error during the reset operations, extending the errors to be non-local, or constructing the noise model from a device's calibration data.
\n", "- Change the circuit from iterated entanglement swapping to iterated applications of a correction circuit from a simple error-correcting code. Do you expect this to be more sensitive to depolarising errors from unitary gates or measurement errors?"]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file diff --git a/examples/expectation_value_example.ipynb b/examples/expectation_value_example.ipynb index 655ce4f6..ba4992dc 100644 --- a/examples/expectation_value_example.ipynb +++ b/examples/expectation_value_example.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Expectation Values"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Given a circuit generating a quantum state $\\lvert \\psi \\rangle$, it is very common to have an operator $H$ and ask for the expectation value $\\langle \\psi \\vert H \\vert \\psi \\rangle$. A notable example is in quantum computational chemistry, where $\\lvert \\psi \\rangle$ encodes the wavefunction for the electronic state of a small molecule, and the energy of the molecule can be derived from the expectation value with respect to the molecule's Hamiltonian operator $H$.
\n", "
\n", "This example uses this chemistry scenario to demonstrate the overall procedure for using `pytket` to perform advanced high-level procedures. We build on top of topics covered by several other example notebooks, including circuit generation, optimisation, and using different backends.
\n", "
\n", "There is limited built-in functionality in `pytket` for obtaining expectation values from circuits. This is designed to encourage users to consider their needs for parallelising the processing of circuits, manipulating results (e.g. filtering, adjusting counts to mitigate errors, and other forms of data processing), or more advanced schemes for grouping the terms of the operator into measurement circuits. For this example, suppose that we want to focus on reducing the queueing time for IBM device backends, and filter our shots to eliminate some detected errors.
\n", "
\n", "This notebook makes use of the Qiskit and ProjectQ backend modules `pytket_qiskit` and `pytket_projectq`, as well as the electronic structure module `openfermion`, all three of which should first be installed via `pip`.
\n", "
\n", "We will start by generating an ansatz and Hamiltonian for the chemical of interest. Here, we are just using a simple model of $\\mathrm{H}_2$ with four qubits representing the occupation of four spin orbitals."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit, Qubit, Bit\n", "from sympy import symbols"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Generate ansatz and Hamiltonian:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ansatz = Circuit()\n", "qubits = ansatz.add_q_register(\"q\", 4)\n", "args = symbols(\"a0 a1 a2 a3 a4 a5 a6 a7\")\n", "for i in range(4):\n", " ansatz.Ry(args[i], qubits[i])\n", "for i in range(3):\n", " ansatz.CX(qubits[i], qubits[i + 1])\n", "for i in range(4):\n", " ansatz.Ry(args[4 + i], qubits[i])\n", "ansatz.measure_all()"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["for command in ansatz:\n", " print(command)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In reality, you would use an expectation value calculation as the objective function for a classical optimisation routine to determine the parameter values for the ground state. For the purposes of this notebook, we will use some predetermined values for the ansatz, already optimised for $\\mathrm{H}_2$."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["arg_values = [\n", " 7.17996183e-02,\n", " 2.95442468e-08,\n", " 1.00000015e00,\n", " 1.00000086e00,\n", " 9.99999826e-01,\n", " 1.00000002e00,\n", " 9.99999954e-01,\n", " 1.13489747e-06,\n", "]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ansatz.symbol_substitution(dict(zip(args, arg_values)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can use for example the openfermion library to express an Hamiltonian as a sum of tensors of paulis."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import openfermion as of"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian = (\n", " -0.0970662681676282 * of.QubitOperator(\"\")\n", " + -0.045302615503799284 * of.QubitOperator(\"X0 X1 Y2 Y3\")\n", " + 0.045302615503799284 * of.QubitOperator(\"X0 Y1 Y2 X3\")\n", " + 0.045302615503799284 * of.QubitOperator(\"Y0 X1 X2 Y3\")\n", " + -0.045302615503799284 * of.QubitOperator(\"Y0 Y1 X2 X3\")\n", " + 0.17141282644776884 * of.QubitOperator(\"Z0\")\n", " + 0.16868898170361213 * of.QubitOperator(\"Z0 Z1\")\n", " + 0.12062523483390425 * of.QubitOperator(\"Z0 Z2\")\n", " + 0.16592785033770352 * of.QubitOperator(\"Z0 Z3\")\n", " + 0.17141282644776884 * of.QubitOperator(\"Z1\")\n", " + 0.16592785033770352 * of.QubitOperator(\"Z1 Z2\")\n", " + 0.12062523483390425 * of.QubitOperator(\"Z1 Z3\")\n", " + -0.22343153690813597 * of.QubitOperator(\"Z2\")\n", " + 0.17441287612261608 * of.QubitOperator(\"Z2 Z3\")\n", " + -0.22343153690813597 * of.QubitOperator(\"Z3\")\n", ")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This can be converted into pytket's QubitPauliOperator type.
\n", "
\n", "The OpenFermion `QubitOperator` class represents the operator by its decomposition into a linear combination of Pauli operators (tensor products of the $I$, $X$, $Y$, and $Z$ matrices).
\n", "
\n", "A `QubitPauliString` is a sparse representation of a Pauli operator with support over some subset of qubits."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["pauli_sym = {\"I\": Pauli.I, \"X\": Pauli.X, \"Y\": Pauli.Y, \"Z\": Pauli.Z}"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qps_from_openfermion(paulis):\n", " \"\"\"Convert OpenFermion tensor of Paulis to pytket QubitPauliString.\"\"\"\n", " qlist = []\n", " plist = []\n", " for q, p in paulis:\n", " qlist.append(Qubit(q))\n", " plist.append(pauli_sym[p])\n", " return QubitPauliString(qlist, plist)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qpo_from_openfermion(openf_op):\n", " \"\"\"Convert OpenFermion QubitOperator to pytket QubitPauliOperator.\"\"\"\n", " tk_op = dict()\n", " for term, coeff in openf_op.terms.items():\n", " string = qps_from_openfermion(term)\n", " tk_op[string] = coeff\n", " return QubitPauliOperator(tk_op)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian_op = qpo_from_openfermion(hamiltonian)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can simulate this exactly using a statevector simulator like ProjectQ. This has a built-in method for fast calculations of expectation values that works well for small examples like this."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.projectq import ProjectQBackend"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = ProjectQBackend()\n", "ideal_energy = backend.get_operator_expectation_value(ansatz, hamiltonian_op)\n", "print(ideal_energy)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Ideally the state generated by this ansatz will only span the computational basis states with exactly two of the four qubits in state $\\lvert 1 \\rangle$. This is because these basis states correspond to two electrons being present in the molecule.
\n", "
\n", "This ansatz is a hardware-efficient model that is designed to explore a large portion of the Hilbert space with relatively few entangling gates. Unfortunately, with this much freedom, it will regularly generate states that have no physical interpretation such as states spanning multiple basis states corresponding to different numbers of electrons in the system (which we assume is fixed and conserved).
\n", "
\n", "We can mitigate this by using a syndrome qubit that calculates the parity of the other qubits. Post-selecting this syndrome with $\\langle 0 \\rvert$ will project the remaining state onto the subspace of basis states with even parity, increasing the likelihood the observed state will be a physically admissible state.
\n", "
\n", "Even if the ansatz parameters are tuned to give a physical state, real devices have noise and imperfect gates, so in practice we may also measure bad states with a small probability. If this syndrome qubit is measured as 1, it means an error has definitely occurred, so we should discard the shot."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["syn = Qubit(\"synq\", 0)\n", "syn_res = Bit(\"synres\", 0)\n", "ansatz.add_qubit(syn)\n", "ansatz.add_bit(syn_res)\n", "for qb in qubits:\n", " ansatz.CX(qb, syn)\n", "ansatz.Measure(syn, syn_res)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Using this, we can define a filter function which removes the shots which the syndrome qubit detected as erroneous. `BackendResult` objects allow retrieval of shots in any bit order, so we can retrieve the `synres` results separately and use them to filter the shots from the remaining bits. The Backends example notebook describes this in more detail."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from collections import Counter"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def filter_shots(backend_result, syn_res_bit):\n", " bits = sorted(backend_result.get_bitlist())\n", " bits.remove(syn_res_bit)\n", " syn_shots = backend_result.get_shots([syn_res])[:, 0]\n", " main_shots = backend_result.get_shots(bits)\n", " return main_shots[syn_shots == 0]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def filter_counts(backend_result, syn_res_bit):\n", " bits = sorted(backend_result.get_bitlist())\n", " syn_index = bits.index(syn_res_bit)\n", " counts = backend_result.get_counts()\n", " filtered_counts = Counter()\n", " for readout, count in counts.items():\n", " if readout[syn_index] == 0:\n", " filtered_readout = tuple(v for i, v in enumerate(readout) if i != syn_index)\n", " filtered_counts[filtered_readout] += count\n", " return filtered_counts"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Depending on which backend we will be using, we will need to compile each circuit we run to conform to the gate set and connectivity constraints. We can define a compilation pass for each backend that optimises the circuit and maps it onto the backend's gate set and connectivity constraints. We don't expect this to change our circuit too much as it is already near-optimal."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.passes import OptimisePhaseGadgets, SequencePass"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def compiler_pass(backend):\n", " return SequencePass([OptimisePhaseGadgets(), backend.default_compilation_pass()])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Given the full statevector, the expectation value can be calculated simply by matrix multiplication. However, with a real quantum system, we cannot observe the full statevector directly. Fortunately, the Pauli decomposition of the operator gives us a sequence of measurements we should apply to obtain the relevant information to reconstruct the expectation value.
\n", "
\n", "The utility method `append_pauli_measurement` takes a single term of a `QubitPauliOperator` (a `QubitPauliString`) and appends measurements in the corresponding bases to obtain the expectation value for that particular Pauli operator. We will want to make a new `Circuit` object for each of the measurements we wish to observe.
\n", ""]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.predicates import CompilationUnit\n", "from pytket.utils import append_pauli_measurement"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def gen_pauli_measurement_circuits(state_circuit, compiler_pass, operator):\n", " # compile main circuit once\n", " state_cu = CompilationUnit(state_circuit)\n", " compiler_pass.apply(state_cu)\n", " compiled_state = state_cu.circuit\n", " final_map = state_cu.final_map\n", " # make a measurement circuit for each pauli\n", " pauli_circuits = []\n", " coeffs = []\n", " energy = 0\n", " for p, c in operator.terms.items():\n", " if p == ():\n", " # constant term\n", " energy += c\n", " else:\n", " # make measurement circuits and compile them\n", " pauli_circ = Circuit(state_circuit.n_qubits - 1) # ignore syndrome qubit\n", " append_pauli_measurement(qps_from_openfermion(p), pauli_circ)\n", " pauli_cu = CompilationUnit(pauli_circ)\n", " compiler_pass.apply(pauli_cu)\n", " pauli_circ = pauli_cu.circuit\n", " init_map = pauli_cu.initial_map\n", " # map measurements onto the placed qubits from the state\n", " rename_map = {\n", " i: final_map[o] for o, i in init_map.items() if o in final_map\n", " }\n", " pauli_circ.rename_units(rename_map)\n", " state_and_measure = compiled_state.copy()\n", " state_and_measure.append(pauli_circ)\n", " pauli_circuits.append(state_and_measure)\n", " coeffs.append(c)\n", " return pauli_circuits, coeffs, energy"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can now start composing these together to get our generalisable expectation value function. Passing all of our circuits to `process_circuits` allows them to be submitted to IBM Quantum devices at the same time, giving substantial savings in overall queueing time. Since the backend will cache any results from `Backend.process_circuits`, we will remove the results when we are done with them to prevent memory bloating when this method is called many times."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils import expectation_from_shots, expectation_from_counts"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def expectation_value(state_circuit, operator, backend, n_shots):\n", " if backend.supports_expectation:\n", " circuit = state_circuit.copy()\n", " compiled_circuit = backend.get_compiled_circuit(circuit)\n", " return backend.get_operator_expectation_value(\n", " compiled_circuit, qpo_from_openfermion(operator)\n", " )\n", " elif backend.supports_shots:\n", " syn_res_index = state_circuit.bit_readout[syn_res]\n", " pauli_circuits, coeffs, energy = gen_pauli_measurement_circuits(\n", " state_circuit, compiler_pass(backend), operator\n", " )\n", " handles = backend.process_circuits(pauli_circuits, n_shots=n_shots)\n", " for handle, coeff in zip(handles, coeffs):\n", " res = backend.get_result(handle)\n", " filtered = filter_shots(res, syn_res)\n", " energy += coeff * expectation_from_shots(filtered)\n", " backend.pop_result(handle)\n", " return energy\n", " elif backend.supports_counts:\n", " syn_res_index = state_circuit.bit_readout[syn_res]\n", " pauli_circuits, coeffs, energy = gen_pauli_measurement_circuits(\n", " state_circuit, compiler_pass(backend), operator\n", " )\n", " handles = backend.process_circuits(pauli_circuits, n_shots=n_shots)\n", " for handle, coeff in zip(handles, coeffs):\n", " res = backend.get_result(handle)\n", " filtered = filter_counts(res, syn_res)\n", " energy += coeff * expectation_from_counts(filtered)\n", " backend.pop_result(handle)\n", " return energy\n", " else:\n", " raise NotImplementedError(\"Implementation for state to be written\")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["...and then run it for our ansatz. `AerBackend` supports faster expectation value from snapshopts (using the `AerBackend.get_operator_expectation_value` method), but this only works when all the qubits in the circuit are default register qubits that go up from 0. So we will need to rename `synq`."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import IBMQEmulatorBackend, AerBackend"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ansatz.rename_units({Qubit(\"synq\", 0): Qubit(\"q\", 4)})"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(expectation_value(ansatz, hamiltonian, AerBackend(), 8000))\n", "# Try replacing IBMQEmulatorBackend with IBMQBackend to submit the circuits to a real IBM Quantum device.\n", "print(expectation_value(ansatz, hamiltonian, IBMQEmulatorBackend(\"ibmq_manila\"), 8000))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["For basic practice with using pytket backends and their results, try editing the code here to:
\n", "* Extend `expectation_value` to work with statevector backends (e.g. `AerStateBackend`)
\n", "* Remove the row filtering from `filter_shots` and see the effect on the expectation value on a noisy simulation/device
\n", "* Adapt `filter_shots` to be able to filter a counts dictionary and adapt `expectation_value` to calulate the result using the counts summary from the backend (`pytket.utils.expectation_from_counts` will be useful here)"]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Expectation values"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Given a circuit generating a quantum state $\\lvert \\psi \\rangle$, it is very common to have an operator $H$ and ask for the expectation value $\\langle \\psi \\vert H \\vert \\psi \\rangle$. A notable example is in quantum computational chemistry, where $\\lvert \\psi \\rangle$ encodes the wavefunction for the electronic state of a small molecule, and the energy of the molecule can be derived from the expectation value with respect to the molecule's Hamiltonian operator $H$.
\n", "
\n", "This example uses this chemistry scenario to demonstrate the overall procedure for using `pytket` to perform advanced high-level procedures. We build on top of topics covered by several other example notebooks, including circuit generation, optimisation, and using different backends.
\n", "
\n", "There is limited built-in functionality in `pytket` for obtaining expectation values from circuits. This is designed to encourage users to consider their needs for parallelising the processing of circuits, manipulating results (e.g. filtering, adjusting counts to mitigate errors, and other forms of data processing), or more advanced schemes for grouping the terms of the operator into measurement circuits. For this example, suppose that we want to focus on reducing the queueing time for IBM device backends, and filter our shots to eliminate some detected errors.
\n", "
\n", "This notebook makes use of the Qiskit and ProjectQ backend modules `pytket_qiskit` and `pytket_projectq`, as well as the electronic structure module `openfermion`, all three of which should first be installed via `pip`.
\n", "
\n", "We will start by generating an ansatz and Hamiltonian for the chemical of interest. Here, we are just using a simple model of $\\mathrm{H}_2$ with four qubits representing the occupation of four spin orbitals."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit, Qubit, Bit\n", "from sympy import symbols"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Generate ansatz and Hamiltonian:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ansatz = Circuit()\n", "qubits = ansatz.add_q_register(\"q\", 4)\n", "args = symbols(\"a0 a1 a2 a3 a4 a5 a6 a7\")\n", "for i in range(4):\n", " ansatz.Ry(args[i], qubits[i])\n", "for i in range(3):\n", " ansatz.CX(qubits[i], qubits[i + 1])\n", "for i in range(4):\n", " ansatz.Ry(args[4 + i], qubits[i])\n", "ansatz.measure_all()"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["for command in ansatz:\n", " print(command)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In reality, you would use an expectation value calculation as the objective function for a classical optimisation routine to determine the parameter values for the ground state. For the purposes of this notebook, we will use some predetermined values for the ansatz, already optimised for $\\mathrm{H}_2$."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["arg_values = [\n", " 7.17996183e-02,\n", " 2.95442468e-08,\n", " 1.00000015e00,\n", " 1.00000086e00,\n", " 9.99999826e-01,\n", " 1.00000002e00,\n", " 9.99999954e-01,\n", " 1.13489747e-06,\n", "]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ansatz.symbol_substitution(dict(zip(args, arg_values)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can use for example the openfermion library to express an Hamiltonian as a sum of tensors of paulis."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import openfermion as of"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian = (\n", " -0.0970662681676282 * of.QubitOperator(\"\")\n", " + -0.045302615503799284 * of.QubitOperator(\"X0 X1 Y2 Y3\")\n", " + 0.045302615503799284 * of.QubitOperator(\"X0 Y1 Y2 X3\")\n", " + 0.045302615503799284 * of.QubitOperator(\"Y0 X1 X2 Y3\")\n", " + -0.045302615503799284 * of.QubitOperator(\"Y0 Y1 X2 X3\")\n", " + 0.17141282644776884 * of.QubitOperator(\"Z0\")\n", " + 0.16868898170361213 * of.QubitOperator(\"Z0 Z1\")\n", " + 0.12062523483390425 * of.QubitOperator(\"Z0 Z2\")\n", " + 0.16592785033770352 * of.QubitOperator(\"Z0 Z3\")\n", " + 0.17141282644776884 * of.QubitOperator(\"Z1\")\n", " + 0.16592785033770352 * of.QubitOperator(\"Z1 Z2\")\n", " + 0.12062523483390425 * of.QubitOperator(\"Z1 Z3\")\n", " + -0.22343153690813597 * of.QubitOperator(\"Z2\")\n", " + 0.17441287612261608 * of.QubitOperator(\"Z2 Z3\")\n", " + -0.22343153690813597 * of.QubitOperator(\"Z3\")\n", ")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This can be converted into pytket's QubitPauliOperator type.
\n", "
\n", "The OpenFermion `QubitOperator` class represents the operator by its decomposition into a linear combination of Pauli operators (tensor products of the $I$, $X$, $Y$, and $Z$ matrices).
\n", "
\n", "A `QubitPauliString` is a sparse representation of a Pauli operator with support over some subset of qubits."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["pauli_sym = {\"I\": Pauli.I, \"X\": Pauli.X, \"Y\": Pauli.Y, \"Z\": Pauli.Z}"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qps_from_openfermion(paulis):\n", " \"\"\"Convert OpenFermion tensor of Paulis to pytket QubitPauliString.\"\"\"\n", " qlist = []\n", " plist = []\n", " for q, p in paulis:\n", " qlist.append(Qubit(q))\n", " plist.append(pauli_sym[p])\n", " return QubitPauliString(qlist, plist)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qpo_from_openfermion(openf_op):\n", " \"\"\"Convert OpenFermion QubitOperator to pytket QubitPauliOperator.\"\"\"\n", " tk_op = dict()\n", " for term, coeff in openf_op.terms.items():\n", " string = qps_from_openfermion(term)\n", " tk_op[string] = coeff\n", " return QubitPauliOperator(tk_op)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian_op = qpo_from_openfermion(hamiltonian)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can simulate this exactly using a statevector simulator like ProjectQ. This has a built-in method for fast calculations of expectation values that works well for small examples like this."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.projectq import ProjectQBackend"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = ProjectQBackend()\n", "ideal_energy = backend.get_operator_expectation_value(ansatz, hamiltonian_op)\n", "print(ideal_energy)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Ideally the state generated by this ansatz will only span the computational basis states with exactly two of the four qubits in state $\\lvert 1 \\rangle$. This is because these basis states correspond to two electrons being present in the molecule.
\n", "
\n", "This ansatz is a hardware-efficient model that is designed to explore a large portion of the Hilbert space with relatively few entangling gates. Unfortunately, with this much freedom, it will regularly generate states that have no physical interpretation such as states spanning multiple basis states corresponding to different numbers of electrons in the system (which we assume is fixed and conserved).
\n", "
\n", "We can mitigate this by using a syndrome qubit that calculates the parity of the other qubits. Post-selecting this syndrome with $\\langle 0 \\rvert$ will project the remaining state onto the subspace of basis states with even parity, increasing the likelihood the observed state will be a physically admissible state.
\n", "
\n", "Even if the ansatz parameters are tuned to give a physical state, real devices have noise and imperfect gates, so in practice we may also measure bad states with a small probability. If this syndrome qubit is measured as 1, it means an error has definitely occurred, so we should discard the shot."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["syn = Qubit(\"synq\", 0)\n", "syn_res = Bit(\"synres\", 0)\n", "ansatz.add_qubit(syn)\n", "ansatz.add_bit(syn_res)\n", "for qb in qubits:\n", " ansatz.CX(qb, syn)\n", "ansatz.Measure(syn, syn_res)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Using this, we can define a filter function which removes the shots which the syndrome qubit detected as erroneous. `BackendResult` objects allow retrieval of shots in any bit order, so we can retrieve the `synres` results separately and use them to filter the shots from the remaining bits. The Backends example notebook describes this in more detail."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from collections import Counter"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def filter_shots(backend_result, syn_res_bit):\n", " bits = sorted(backend_result.get_bitlist())\n", " bits.remove(syn_res_bit)\n", " syn_shots = backend_result.get_shots([syn_res])[:, 0]\n", " main_shots = backend_result.get_shots(bits)\n", " return main_shots[syn_shots == 0]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def filter_counts(backend_result, syn_res_bit):\n", " bits = sorted(backend_result.get_bitlist())\n", " syn_index = bits.index(syn_res_bit)\n", " counts = backend_result.get_counts()\n", " filtered_counts = Counter()\n", " for readout, count in counts.items():\n", " if readout[syn_index] == 0:\n", " filtered_readout = tuple(v for i, v in enumerate(readout) if i != syn_index)\n", " filtered_counts[filtered_readout] += count\n", " return filtered_counts"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Depending on which backend we will be using, we will need to compile each circuit we run to conform to the gate set and connectivity constraints. We can define a compilation pass for each backend that optimises the circuit and maps it onto the backend's gate set and connectivity constraints. We don't expect this to change our circuit too much as it is already near-optimal."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.passes import OptimisePhaseGadgets, SequencePass"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def compiler_pass(backend):\n", " return SequencePass([OptimisePhaseGadgets(), backend.default_compilation_pass()])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Given the full statevector, the expectation value can be calculated simply by matrix multiplication. However, with a real quantum system, we cannot observe the full statevector directly. Fortunately, the Pauli decomposition of the operator gives us a sequence of measurements we should apply to obtain the relevant information to reconstruct the expectation value.
\n", "
\n", "The utility method `append_pauli_measurement` takes a single term of a `QubitPauliOperator` (a `QubitPauliString`) and appends measurements in the corresponding bases to obtain the expectation value for that particular Pauli operator. We will want to make a new `Circuit` object for each of the measurements we wish to observe.
\n", ""]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.predicates import CompilationUnit\n", "from pytket.utils import append_pauli_measurement"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def gen_pauli_measurement_circuits(state_circuit, compiler_pass, operator):\n", " # compile main circuit once\n", " state_cu = CompilationUnit(state_circuit)\n", " compiler_pass.apply(state_cu)\n", " compiled_state = state_cu.circuit\n", " final_map = state_cu.final_map\n", " # make a measurement circuit for each pauli\n", " pauli_circuits = []\n", " coeffs = []\n", " energy = 0\n", " for p, c in operator.terms.items():\n", " if p == ():\n", " # constant term\n", " energy += c\n", " else:\n", " # make measurement circuits and compile them\n", " pauli_circ = Circuit(state_circuit.n_qubits - 1) # ignore syndrome qubit\n", " append_pauli_measurement(qps_from_openfermion(p), pauli_circ)\n", " pauli_cu = CompilationUnit(pauli_circ)\n", " compiler_pass.apply(pauli_cu)\n", " pauli_circ = pauli_cu.circuit\n", " init_map = pauli_cu.initial_map\n", " # map measurements onto the placed qubits from the state\n", " rename_map = {\n", " i: final_map[o] for o, i in init_map.items() if o in final_map\n", " }\n", " pauli_circ.rename_units(rename_map)\n", " state_and_measure = compiled_state.copy()\n", " state_and_measure.append(pauli_circ)\n", " pauli_circuits.append(state_and_measure)\n", " coeffs.append(c)\n", " return pauli_circuits, coeffs, energy"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can now start composing these together to get our generalisable expectation value function. Passing all of our circuits to `process_circuits` allows them to be submitted to IBM Quantum devices at the same time, giving substantial savings in overall queueing time. Since the backend will cache any results from `Backend.process_circuits`, we will remove the results when we are done with them to prevent memory bloating when this method is called many times."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils import expectation_from_shots, expectation_from_counts"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def expectation_value(state_circuit, operator, backend, n_shots):\n", " if backend.supports_expectation:\n", " circuit = state_circuit.copy()\n", " compiled_circuit = backend.get_compiled_circuit(circuit)\n", " return backend.get_operator_expectation_value(\n", " compiled_circuit, qpo_from_openfermion(operator)\n", " )\n", " elif backend.supports_shots:\n", " syn_res_index = state_circuit.bit_readout[syn_res]\n", " pauli_circuits, coeffs, energy = gen_pauli_measurement_circuits(\n", " state_circuit, compiler_pass(backend), operator\n", " )\n", " handles = backend.process_circuits(pauli_circuits, n_shots=n_shots)\n", " for handle, coeff in zip(handles, coeffs):\n", " res = backend.get_result(handle)\n", " filtered = filter_shots(res, syn_res)\n", " energy += coeff * expectation_from_shots(filtered)\n", " backend.pop_result(handle)\n", " return energy\n", " elif backend.supports_counts:\n", " syn_res_index = state_circuit.bit_readout[syn_res]\n", " pauli_circuits, coeffs, energy = gen_pauli_measurement_circuits(\n", " state_circuit, compiler_pass(backend), operator\n", " )\n", " handles = backend.process_circuits(pauli_circuits, n_shots=n_shots)\n", " for handle, coeff in zip(handles, coeffs):\n", " res = backend.get_result(handle)\n", " filtered = filter_counts(res, syn_res)\n", " energy += coeff * expectation_from_counts(filtered)\n", " backend.pop_result(handle)\n", " return energy\n", " else:\n", " raise NotImplementedError(\"Implementation for state to be written\")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["...and then run it for our ansatz. `AerBackend` supports faster expectation value from snapshopts (using the `AerBackend.get_operator_expectation_value` method), but this only works when all the qubits in the circuit are default register qubits that go up from 0. So we will need to rename `synq`."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import IBMQEmulatorBackend, AerBackend"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ansatz.rename_units({Qubit(\"synq\", 0): Qubit(\"q\", 4)})"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(expectation_value(ansatz, hamiltonian, AerBackend(), 8000))\n", "# Try replacing IBMQEmulatorBackend with IBMQBackend to submit the circuits to a real IBM Quantum device.\n", "print(expectation_value(ansatz, hamiltonian, IBMQEmulatorBackend(\"ibmq_manila\"), 8000))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["For basic practice with using pytket backends and their results, try editing the code here to:
\n", "* Extend `expectation_value` to work with statevector backends (e.g. `AerStateBackend`)
\n", "* Remove the row filtering from `filter_shots` and see the effect on the expectation value on a noisy simulation/device
\n", "* Adapt `filter_shots` to be able to filter a counts dictionary and adapt `expectation_value` to calulate the result using the counts summary from the backend (`pytket.utils.expectation_from_counts` will be useful here)"]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file diff --git a/examples/mapping_example.ipynb b/examples/mapping_example.ipynb index bbb4001b..28d8e4d2 100644 --- a/examples/mapping_example.ipynb +++ b/examples/mapping_example.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Qubit Mapping and Routing"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In this tutorial we will show how the problem of mapping from logical quantum circuits to physically permitted circuits is solved automatically in TKET. The basic examples require only the installation of pytket, ```pip install pytket```."]}, {"cell_type": "markdown", "metadata": {}, "source": ["There is a wide variety of different blueprints for realising quantum computers, including the well known superconducting and ion trap devices. Different devices come with different constraints, such as a limited primitive gate set for universal quantum computing. Often this limited gate set accommodates an additional constraint, that two-qubit gates can not be executed between all pairs of qubits."]}, {"cell_type": "markdown", "metadata": {}, "source": ["In software, typically this constraint is presented as a \"connectivity\" graph where vertices connected by an edge represents pairs of physical qubits which two-qubit gates can be executed on. As programmers usually write logical quantum circuits with no sense of architecture (or may want to run their circuit on a range of hardware with different connectivity constraints), most quantum software development kits offer the means to automatically solve this constraint. One common way is to automatically add logical ```SWAP``` gates to a Circuit, changing the position of logical qubits on physical qubits until a two-qubit gate can be realised. This is an active area of research in quantum computing and a problem we discuss in our paper \"On The Qubit Routing Problem\" - arXiv:1902.08091."]}, {"cell_type": "markdown", "metadata": {}, "source": ["In TKET this constraint is represented by the ```Architecture``` class. An Architecture object requires a coupling map to be created, a list of pairs of qubits which defines where two-qubit primitives may be executed. A coupling map can be produced naively by the integer indexing of nodes and edges in some architecture."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.architecture import Architecture\n", "from pytket.circuit import Node"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import networkx as nx\n", "from typing import List, Union, Tuple"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def draw_graph(coupling_map: List[Union[Tuple[int, int], Tuple[Node, Node]]]):\n", " coupling_graph = nx.Graph(coupling_map)\n", " nx.draw(coupling_graph, labels={node: node for node in coupling_graph.nodes()})"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["simple_coupling_map = [(0, 1), (1, 2), (2, 3)]\n", "simple_architecture = Architecture(simple_coupling_map)\n", "draw_graph(simple_coupling_map)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Alternatively we could use the `Node` class to assign our nodes - you will see why this can be helpful later. Lets create an Architecture with an identical graph:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["node_0 = Node(\"e0\", 0)\n", "node_1 = Node(\"e1\", 1)\n", "node_2 = Node(\"e2\", 2)\n", "node_3 = Node(\"e3\", 3)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["id_coupling_map = [(node_0, node_1), (node_1, node_2), (node_2, node_3)]\n", "id_architecture = Architecture(id_coupling_map)\n", "draw_graph(id_coupling_map)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can also create an ID with an arbitrary-dimensional index. Let us make a 2x2x2 cube:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["node_000 = Node(\"cube\", [0, 0, 0])\n", "node_001 = Node(\"cube\", [0, 0, 1])\n", "node_010 = Node(\"cube\", [0, 1, 0])\n", "node_011 = Node(\"cube\", [0, 1, 1])\n", "node_100 = Node(\"cube\", [1, 0, 0])\n", "node_101 = Node(\"cube\", [1, 0, 1])\n", "node_110 = Node(\"cube\", [1, 1, 0])\n", "node_111 = Node(\"cube\", [1, 1, 1])"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["cube_coupling_map = [\n", " (node_000, node_001),\n", " (node_000, node_010),\n", " (node_010, node_011),\n", " (node_001, node_011),\n", " (node_000, node_100),\n", " (node_001, node_101),\n", " (node_010, node_110),\n", " (node_011, node_111),\n", " (node_100, node_101),\n", " (node_100, node_110),\n", " (node_110, node_111),\n", " (node_101, node_111),\n", "]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["cube_architecture = Architecture(cube_coupling_map)\n", "draw_graph(cube_coupling_map)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["To avoid that tedium though we could just use our SquareGrid Architecture:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.architecture import SquareGrid"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["alternative_cube_architecture = SquareGrid(2, 2, 2)\n", "draw_graph(alternative_cube_architecture.coupling)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The current range of quantum computers are commonly referred to as Noisy-Intermediate-Scale-Quantum devices i.e. NISQ devices. The impact of noise is a primary concern during compilation and incentivizes producing physically permitted circuits that have a minimal number of gates. For this reason benchmarking in this area is often completed by comparing the final number of two-qubit (or particularly SWAP gates) in compiled circuits."]}, {"cell_type": "markdown", "metadata": {}, "source": ["However it is important to remember that adding logical SWAP gates to minimise gate count is not the only way this constraint can be met, with large scale architecture-aware synthesis methods and fidelity aware methods amongst other approaches producing viable physically permitted circuits. It is likely that no SINGLE approach is better for all circuits, but the ability to use different approaches where best fitted will give the best results during compilation."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Producing physically valid circuits is completed via the `MappingManager` class, which aims to accomodate a wide range of approaches."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.mapping import MappingManager"]}, {"cell_type": "markdown", "metadata": {}, "source": ["A `MappingManager` object requires an `Architecture` object at construction."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["mapping_manager = MappingManager(id_architecture)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["All mapping is done through the `MappingManager.route_circuit` method. The `MappingManager.route_circuit` method has two arguments, the first a Circuit to be routed (which is mutated), the second a `List[RoutingMethodCircuit]` object that defines how the mapping is completed."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Later we will look at defining our own `RoutingMethodCircuit` objects, but initially lets consider one thats already available."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.mapping import LexiLabellingMethod, LexiRouteRoutingMethod"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["lexi_label = LexiLabellingMethod()\n", "lexi_route = LexiRouteRoutingMethod(10)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The `lexi_route` object here is of little use outside `MappingManager`. Note that it takes a lookahead parameter, which will affect the performance of the method, defining the number of two-qubit gates it considers when finding `SWAP` gates to add."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit, OpType\n", "from pytket.circuit import display"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = (\n", " Circuit(4)\n", " .CX(0, 1)\n", " .CX(1, 2)\n", " .CX(0, 2)\n", " .CX(0, 3)\n", " .CX(2, 3)\n", " .CX(1, 3)\n", " .CX(0, 1)\n", " .measure_all()\n", ")\n", "display.render_circuit_jupyter(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can also look at which logical qubits are interacting."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils import Graph"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["Graph(c).get_qubit_graph()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["By running the `MappingManager.route_circuit` method on our circuit `c` with the `LexiLabellingMethod` and `LexiRouteRoutingMethod` objects as an argument, qubits in `c` with some physical requirements will be relabelled and the qubit graph modified (by the addition of SWAP gates and relabelling some CX as BRIDGE gates) such that the qubit graph is isomorphic to some subgraph of the full architecture."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["mapping_manager.route_circuit(c, [lexi_label, lexi_route])\n", "display.render_circuit_jupyter(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The graph:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["Graph(c).get_qubit_graph()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The resulting circuit may also change if we reduce the lookahead parameter."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = (\n", " Circuit(4)\n", " .CX(0, 1)\n", " .CX(1, 2)\n", " .CX(0, 2)\n", " .CX(0, 3)\n", " .CX(2, 3)\n", " .CX(1, 3)\n", " .CX(0, 1)\n", " .measure_all()\n", ")\n", "mapping_manager.route_circuit(c, [lexi_label, LexiRouteRoutingMethod(1)])\n", "display.render_circuit_jupyter(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can also pass multiple `RoutingMethod` options for Routing in a ranked List. Each `RoutingMethod` option has a function for checking whether it can usefully modify a subcircuit at a stage in Routing. To choose, each method in the List is checked in order until one returns True. This will be discussed more later."]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can aid the mapping procedure by relabelling qubits in advance. This can be completed using the `Placement` class."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.placement import Placement, LinePlacement, GraphPlacement"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The default ```Placement``` assigns logical qubits to physical qubits as they are encountered during routing. ```LinePlacement``` uses a strategy described in https://arxiv.org/abs/1902.08091. ```GraphPlacement``` is described in Section 7.1 of https://arxiv.org/abs/2003.10611. Lets look at how we can use the ```LinePlacement``` class.`"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["line_placement = LinePlacement(id_architecture)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = (\n", " Circuit(4)\n", " .CX(0, 1)\n", " .CX(1, 2)\n", " .CX(0, 2)\n", " .CX(0, 3)\n", " .CX(2, 3)\n", " .CX(1, 3)\n", " .CX(0, 1)\n", " .measure_all()\n", ")\n", "line_placement.place(c)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["display.render_circuit_jupyter(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Note that one qubit remains unplaced in this example. `LexiRouteRoutingMethod` will dynamically assign it during mapping."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Different placements will lead to different selections of SWAP gates being added. However each different routed circuit will preserve the original unitary action of the full circuit while respecting connectivity constraints."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["mapping_manager.route_circuit(c, [lexi_label, lexi_route])\n", "display.render_circuit_jupyter(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The graph:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["Graph(c).get_qubit_graph()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["However, small changes to the depth of lookahead or the original assignment of `Architecture` `Node` can greatly affect the resulting physical circuit for the `LexiRouteRoutingMethod` method. Considering this variance, it should be possible to easily throw additional computational resources at the problem if necessary, which is something TKET is leaning towards with the ability to define custom `RoutingCircuitMethod` objects."]}, {"cell_type": "markdown", "metadata": {}, "source": ["To define a new `RoutingMethodCircuit` method though, we first need to understand how it is used in `MappingManager` and routing. The `MappingManager.route_circuit` method treats the global problem of mapping to physical circuits as many sequential sub-problems. Consider the following problem."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.placement import place_with_map"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ = Circuit(4).CX(0, 1).CX(1, 2).CX(0, 2).CX(0, 3).CX(2, 3).CX(1, 3).CX(0, 1)\n", "naive_map = {\n", " circ.qubits[0]: node_0,\n", " circ.qubits[1]: node_1,\n", " circ.qubits[2]: node_2,\n", " circ.qubits[3]: node_3,\n", "}\n", "place_with_map(circ, naive_map)\n", "Graph(circ).get_DAG()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["So what happens when we run the following?"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["mapping_manager.route_circuit(circ, [lexi_route])\n", "Graph(circ).get_DAG()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Sequential mapping typically works by partitioning the circuit into two, a first partition comprising a connected subcircuit that is physically permitted, a second partition that is not. Therefore, the first thing `MappingManager.route_circuit` does is find this partition for the passed circuit, by iterating through gates in the circuit."]}, {"cell_type": "markdown", "metadata": {}, "source": ["We will construct the partitions ourselves for illustrative purposes. Lets assume we are routing for the four qubit line architecture (qubits are connected to adjacent indices) \"simple_architecture\" we constructed earlier."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ_first_partition = Circuit(4).CX(0, 1).CX(1, 2)\n", "place_with_map(circ_first_partition, naive_map)\n", "Graph(circ_first_partition).get_DAG()"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ_second_partition = Circuit(4).CX(0, 2).CX(0, 3).CX(2, 3).CX(1, 3).CX(0, 1)\n", "place_with_map(circ_second_partition, naive_map)\n", "Graph(circ_second_partition).get_DAG()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Note that there are gates in the second partition that would be physically permitted, if they were not dependent on other gates that are not."]}, {"cell_type": "markdown", "metadata": {}, "source": ["The next step is to modify the second partition circuit to move it closer being physically permitted. Here the `LexiRouteRoutingMethod` as before will either insert a SWAP gate at the start of the partition, or will substitute a CX gate in the first slice of the partition with a BRIDGE gate."]}, {"cell_type": "markdown", "metadata": {}, "source": ["The option taken by `LexiRouteRoutingethod(1)` is to insert a SWAP gate between the first two nodes of the architecture, swapping their logical states. How does this change the second partition circuit?"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ_second_partition = (\n", " Circuit(4).SWAP(0, 1).CX(1, 2).CX(1, 3).CX(2, 3).CX(0, 3).CX(1, 0)\n", ")\n", "place_with_map(circ_second_partition, naive_map)\n", "Graph(circ_second_partition).get_DAG()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Leaving the full circuit as:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["full_circuit = (\n", " Circuit(4).CX(0, 1).CX(1, 2).SWAP(0, 1).CX(1, 2).CX(1, 3).CX(2, 3).CX(0, 3).CX(1, 0)\n", ")\n", "place_with_map(full_circuit, naive_map)\n", "Graph(full_circuit).get_DAG()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["After a modification is made the partition is updated."]}, {"cell_type": "markdown", "metadata": {}, "source": ["The first partition:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ_first_partition = Circuit(4).CX(0, 1).CX(1, 2).SWAP(0, 1).CX(1, 2)\n", "place_with_map(circ_first_partition, naive_map)\n", "Graph(circ_first_partition).get_DAG()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The second partition:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ_second_partition = Circuit(4).CX(1, 3).CX(2, 3).CX(0, 3).CX(1, 0)\n", "place_with_map(circ_second_partition, naive_map)\n", "Graph(circ_second_partition).get_DAG()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This pattern of modification and upating the partition is repeated until the partition has reached the end of the circuit, i.e. the back side of the partition has no gates in it. Also note that the process of updating the partition has been simplified for this example with \"physically permitted\" encapsulating two-qubit gate constraints only - in the future we expect other arity gates to provide constraints that need to be met. Also note that any modification to the second circuit can willfully modify the qubit labelling and a token swapping network will be automatically added to conform to the new labelling."]}, {"cell_type": "markdown", "metadata": {}, "source": ["We now enough about how `MappingManager` works to add our own `RoutingMethodCircuit`. While `LexiRouteRoutingMethod` is implemented in c++ TKET, giving it some advantages, via lambda functions we can define our own `RoutingMethodCircuit` in python."]}, {"cell_type": "markdown", "metadata": {}, "source": ["A python defined `RoutingMethodCircuit` requires three arguments. The first is a function that given a Circuit (the circuit after the partition) and an Architecture, returns a bool (determining whether the new circuit should be substitued in a full routing process), a new Circuit (a modification of the original circuit such as an added SWAP) a Dict between qubits reflecting any relabelling done in the method, and a Dict between qubits giving any implicit permutation of qubits (such as by adding a SWAP). For some clarity (we will write an example later), lets look at an example function declaration."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from typing import Dict"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def route_subcircuit_func(\n", " circuit: Circuit, architecture: Architecture\n", ") -> Tuple[bool, Circuit, Dict[Node, Node], Dict[Node, Node]]:\n", " return ()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The first return is a bool which detemrines if a given `RoutingMethodCircuit` is suitable for providing a solution at a given partition. `MappingManager.route_circuit` accepts a List of of `RoutingMethod` defining how solutions are found. At the point the partition circuit is modified, the circuit is passed to `RoutingMethodCircuit.routing_method` which additionally to finding a subcircuit substitution, should determine whether it can or can't helpfully modify the partition boundary circuit, and return True if it can. The first `RoutingMethodCircuit` to return True is then used for modification - meaning the ordering of List elements is important."]}, {"cell_type": "markdown", "metadata": {}, "source": ["The third argument sets the maximum number of gates given in the passed Circuit and the fourth argument sets the maximum depth in the passed Circuit."]}, {"cell_type": "markdown", "metadata": {}, "source": ["`LexiRouteRoutingMethod` will always return True, because it can always find some helpful SWAP to insert, and it can dynamically assign logical to physical qubits. Given this, lets construct a more specialised modification - an architecture-aware decomposition of a distance-2 CRy gate. Lets write our function type declarations for each method:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def distance2_CRy_decomp(\n", " circuit: Circuit, architecture: Architecture\n", ") -> Tuple[bool, Circuit, Dict[Node, Node], Dict[Node, Node]]:\n", " return (False, Circuit(), {}, {})"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Where do we start? Lets define a simple scope for our solution: for a single gate in the passed circuit (the circuit after the partition) that has OpType CRy, if the two qubits it's acting on are at distance 2 on the architecture, decompose the gate using BRIDGE gates."]}, {"cell_type": "markdown", "metadata": {}, "source": ["The first restriction is to only have a single gate from the first slice - we can achieve this by setting both the maximum depth and size parameters to 1."]}, {"cell_type": "markdown", "metadata": {}, "source": ["The second restriction is for the gate to have OpType CRy and for the qubits to be at distance 2 - we can check this restriction in a `distance2_CRy_check` method."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def distance2_CRy_check(circuit: Circuit, architecture: Architecture) -> bool:\n", " if circuit.n_gates != 1:\n", " raise ValueError(\n", " \"Circuit for CRy check should only have 1 gate, please change parameters of method declaration.\"\n", " )\n", " command = circuit.get_commands()[0]\n", " if command.op.type == OpType.CRy:\n", " # Architecture stores qubits under `Node` identifier\n", " n0 = Node(command.qubits[0].reg_name, command.qubits[0].index)\n", " n1 = Node(command.qubits[1].reg_name, command.qubits[1].index)\n", " # qubits could not be placed in circuit, so check before finding distance\n", " if n0 in architecture.nodes and n1 in architecture.nodes:\n", " # means we can run the decomposition\n", " if architecture.get_distance(n0, n1) == 2:\n", " return True\n", " return False"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The `distance2_CRy_check` confirms whether the required restrictions are respected. Given this, if the `distance2_CRy_decomp` method is called we know where to add the decomposition."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def distance2_CRy_decomp(\n", " circuit: Circuit, architecture: Architecture\n", ") -> Tuple[bool, Circuit, Dict[Node, Node], Dict[Node, Node]]:\n", " worthwhile_substitution = distance2_CRy_check(circuit, architecture)\n", " if worthwhile_substitution == False:\n", " return (False, Circuit(), {}, {})\n", " command = circuit.get_commands()[0]\n", " qubits = command.qubits\n", " # Architecture stores qubits under `Node` identifier\n", " n0 = Node(qubits[0].reg_name, qubits[0].index)\n", " n1 = Node(qubits[1].reg_name, qubits[1].index)\n\n", " # need to find connecting node for decomposition\n", " adjacent_nodes_0 = architecture.get_adjacent_nodes(n0)\n", " adjacent_nodes_1 = architecture.get_adjacent_nodes(n1)\n", " connecting_nodes = adjacent_nodes_0.intersection(adjacent_nodes_1)\n", " if len(connecting_nodes) == 0:\n", " raise ValueError(\"Qubits for distance-2 CRy decomp are not at distance 2.\")\n", " connecting_node = connecting_nodes.pop()\n", " c = Circuit()\n\n", " # the \"relabelling map\" empty, and the permutation map is qubit to qubit, so add here\n", " permutation_map = dict()\n", " for q in circuit.qubits:\n", " permutation_map[q] = q\n", " c.add_qubit(q)\n", " # rotation, can assume only parameter as CRy\n", " angle = command.op.params[0]\n", " c.Ry(angle, qubits[1])\n", " # distance-2 CX decomp\n", " c.CX(qubits[0], connecting_node).CX(connecting_node, qubits[1])\n", " c.CX(qubits[0], connecting_node).CX(connecting_node, qubits[1])\n", " # rotation\n", " c.Ry(-1 * angle, qubits[1])\n", " # distance-2 CX decomp\n", " c.CX(qubits[0], connecting_node).CX(connecting_node, qubits[1])\n", " c.CX(qubits[0], connecting_node).CX(connecting_node, qubits[1])\n\n", " # the \"relabelling map\" is just qubit to qubit\n", " return (True, c, {}, permutation_map)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Before turning this into a `RoutingMethod` we can try it ourselves."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_c = Circuit(4)\n", "test_c.CRy(0.6, 0, 2)\n", "place_with_map(test_c, naive_map)\n", "Graph(test_c).get_DAG()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["As we can see, our circuit has one CRy gate at distance two away."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(distance2_CRy_check(test_c, id_architecture))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Our method returns True, as expected! We should also test cases where it returns errors or False."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_c_false = Circuit(4)\n", "test_c_false.CRy(0.4, 0, 1)\n", "place_with_map(test_c_false, naive_map)\n", "print(distance2_CRy_check(test_c_false, id_architecture))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_c_error = Circuit(4)\n", "test_c_error.CRy(0.6, 0, 2)\n", "test_c_error.CRy(0.4, 0, 1)\n", "place_with_map(test_c_error, naive_map)\n", "try:\n", " distance2_CRy_check(test_c_error, id_architecture)\n", "except ValueError:\n", " print(\"Error reached!\")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Does the decomposition work?"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_c = Circuit(4)\n", "test_c.CRy(0.6, 0, 2)\n", "place_with_map(test_c, naive_map)\n", "decomp = distance2_CRy_decomp(test_c, id_architecture)\n", "display.render_circuit_jupyter(decomp[1])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Great! Our check function and decomposition method are both working. Lets wrap them into a `RoutingMethodCircuit` and try them out."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.mapping import RoutingMethodCircuit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["cry_rmc = RoutingMethodCircuit(distance2_CRy_decomp, 1, 1)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can use our original `MappingManager` object as it is defined for the same architecture. Lets try it out on a range of circumstances."]}, {"cell_type": "markdown", "metadata": {}, "source": ["If we pass it a full CX circuit without `LexiRouteRoutingMethod`, we should find that `MappingManager` throws an error, as none of the passed methods can route for the given circuit."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = (\n", " Circuit(4)\n", " .CX(0, 1)\n", " .CX(1, 2)\n", " .CX(0, 2)\n", " .CX(0, 3)\n", " .CX(2, 3)\n", " .CX(1, 3)\n", " .CX(0, 1)\n", " .measure_all()\n", ")\n", "place_with_map(c, naive_map)\n", "try:\n", " mapping_manager.route_circuit(c, [cry_rmc])\n", "except RuntimeError:\n", " print(\"Error reached!\")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Alternatively, we can add `LexiRouteRoutingMethod` on top:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = (\n", " Circuit(4)\n", " .CX(0, 1)\n", " .CX(1, 2)\n", " .CX(0, 2)\n", " .CX(0, 3)\n", " .CX(2, 3)\n", " .CX(1, 3)\n", " .CX(0, 1)\n", " .measure_all()\n", ")\n", "place_with_map(c, naive_map)\n", "mapping_manager.route_circuit(c, [cry_rmc, LexiRouteRoutingMethod(10)])\n", "display.render_circuit_jupyter(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["However as there are no CRy gates our new method is unused. We can add one:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = (\n", " Circuit(4)\n", " .CRy(0.6, 0, 2)\n", " .CX(0, 1)\n", " .CX(1, 2)\n", " .CX(0, 2)\n", " .CX(0, 3)\n", " .CX(2, 3)\n", " .CX(1, 3)\n", " .CX(0, 1)\n", " .measure_all()\n", ")\n", "mapping_manager.route_circuit(c, [lexi_label, cry_rmc, LexiRouteRoutingMethod(10)])\n", "display.render_circuit_jupyter(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This time we can see our decomposition! If we reorder the methods though `LexiRouteRoutingMethod` is checked first (and returns True), so our new method is unused. The order is important!"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Finally, lets see what happens if the gate is not at the right distance initially."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = (\n", " Circuit(4)\n", " .CRy(0.6, 0, 3)\n", " .CX(0, 1)\n", " .CX(1, 2)\n", " .CX(0, 2)\n", " .CX(0, 3)\n", " .CX(2, 3)\n", " .CX(1, 3)\n", " .CX(0, 1)\n", " .measure_all()\n", ")\n", "mapping_manager.route_circuit(c, [lexi_label, cry_rmc, LexiRouteRoutingMethod(10)])\n", "display.render_circuit_jupyter(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Above a SWAP gate is inserted by `LexiRouteRoutingMethod` before anything else."]}, {"cell_type": "markdown", "metadata": {}, "source": ["For anyone interested, a simple extension exercise could be to extend this to additionally work for distance-2 CRx and CRz. Alternatively one could improve on the method itself - this approach always decomposes a CRy at distance-2, but is this a good idea?"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Also note that higher performance solutions are coded straight into the TKET c++ codebase. This provides advantages, including that Circuit construction and substitution is unncessary (as with python) as the circuit can be directly modified, however the ability to produce prototypes at the python level is very helpful. If you have a great python implementation but are finding some runtime bottlenecks, why not try implementing it straight into TKET (the code is open source at https://github.com/CQCL/tket)."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Besides the `LexiRouteRoutingMethod()` and the `LexiLabellingMethod()` there are other routing methods in pytket, such as the `AASRouteRoutingMethod()` and the corresponding `AASLabellingMethod()`, which are used to route phase-polynomial boxes using architecture-aware synthesis. Usually circuits contain non-phase-polynomial operations as well, so it is a good idea to combine them with the `LexiRouteRoutingMethod()`, as in the following example:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.mapping import AASRouteRoutingMethod, AASLabellingMethod\n", "from pytket.circuit import PhasePolyBox, Qubit\n", "import numpy as np"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3, 3)\n", "n_qb = 3\n", "qubit_indices = {Qubit(0): 0, Qubit(1): 1, Qubit(2): 2}\n", "phase_polynomial = {(True, False, True): 0.333, (False, False, True): 0.05}\n", "linear_transformation = np.array([[1, 0, 0], [0, 1, 0], [0, 0, 1]])\n", "p_box = PhasePolyBox(n_qb, qubit_indices, phase_polynomial, linear_transformation)\n", "c.add_phasepolybox(p_box, [0, 1, 2])\n", "c.CX(0, 1).CX(0, 2).CX(1, 2)\n", "display.render_circuit_jupyter(c)\n", "nodes = [Node(\"test\", 0), Node(\"test\", 1), Node(\"test\", 2)]\n", "arch = Architecture([[nodes[0], nodes[1]], [nodes[1], nodes[2]]])\n", "mm = MappingManager(arch)\n", "mm.route_circuit(\n", " c,\n", " [\n", " AASRouteRoutingMethod(1),\n", " LexiLabellingMethod(),\n", " LexiRouteRoutingMethod(),\n", " AASLabellingMethod(),\n", " ],\n", ")\n", "display.render_circuit_jupyter(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In this case the order of the methods is not very relevant, because in each step of the routing only one of the methods is suitable. In the first part of the circuit the mapping is done without inserting swaps by the AAS method; in the second part one swap gate is added to the circuit."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Qubit mapping and routing"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In this tutorial we will show how the problem of mapping from logical quantum circuits to physically permitted circuits is solved automatically in TKET. The basic examples require only the installation of pytket, ```pip install pytket```."]}, {"cell_type": "markdown", "metadata": {}, "source": ["There is a wide variety of different blueprints for realising quantum computers, including the well known superconducting and ion trap devices. Different devices come with different constraints, such as a limited primitive gate set for universal quantum computing. Often this limited gate set accommodates an additional constraint, that two-qubit gates can not be executed between all pairs of qubits."]}, {"cell_type": "markdown", "metadata": {}, "source": ["In software, typically this constraint is presented as a \"connectivity\" graph where vertices connected by an edge represents pairs of physical qubits which two-qubit gates can be executed on. As programmers usually write logical quantum circuits with no sense of architecture (or may want to run their circuit on a range of hardware with different connectivity constraints), most quantum software development kits offer the means to automatically solve this constraint. One common way is to automatically add logical ```SWAP``` gates to a Circuit, changing the position of logical qubits on physical qubits until a two-qubit gate can be realised. This is an active area of research in quantum computing and a problem we discuss in our paper \"On The Qubit Routing Problem\" - arXiv:1902.08091."]}, {"cell_type": "markdown", "metadata": {}, "source": ["In TKET this constraint is represented by the ```Architecture``` class. An Architecture object requires a coupling map to be created, a list of pairs of qubits which defines where two-qubit primitives may be executed. A coupling map can be produced naively by the integer indexing of nodes and edges in some architecture."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.architecture import Architecture\n", "from pytket.circuit import Node"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import networkx as nx\n", "from typing import List, Union, Tuple"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def draw_graph(coupling_map: List[Union[Tuple[int, int], Tuple[Node, Node]]]):\n", " coupling_graph = nx.Graph(coupling_map)\n", " nx.draw(coupling_graph, labels={node: node for node in coupling_graph.nodes()})"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["simple_coupling_map = [(0, 1), (1, 2), (2, 3)]\n", "simple_architecture = Architecture(simple_coupling_map)\n", "draw_graph(simple_coupling_map)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Alternatively we could use the `Node` class to assign our nodes - you will see why this can be helpful later. Lets create an Architecture with an identical graph:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["node_0 = Node(\"e0\", 0)\n", "node_1 = Node(\"e1\", 1)\n", "node_2 = Node(\"e2\", 2)\n", "node_3 = Node(\"e3\", 3)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["id_coupling_map = [(node_0, node_1), (node_1, node_2), (node_2, node_3)]\n", "id_architecture = Architecture(id_coupling_map)\n", "draw_graph(id_coupling_map)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can also create an ID with an arbitrary-dimensional index. Let us make a 2x2x2 cube:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["node_000 = Node(\"cube\", [0, 0, 0])\n", "node_001 = Node(\"cube\", [0, 0, 1])\n", "node_010 = Node(\"cube\", [0, 1, 0])\n", "node_011 = Node(\"cube\", [0, 1, 1])\n", "node_100 = Node(\"cube\", [1, 0, 0])\n", "node_101 = Node(\"cube\", [1, 0, 1])\n", "node_110 = Node(\"cube\", [1, 1, 0])\n", "node_111 = Node(\"cube\", [1, 1, 1])"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["cube_coupling_map = [\n", " (node_000, node_001),\n", " (node_000, node_010),\n", " (node_010, node_011),\n", " (node_001, node_011),\n", " (node_000, node_100),\n", " (node_001, node_101),\n", " (node_010, node_110),\n", " (node_011, node_111),\n", " (node_100, node_101),\n", " (node_100, node_110),\n", " (node_110, node_111),\n", " (node_101, node_111),\n", "]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["cube_architecture = Architecture(cube_coupling_map)\n", "draw_graph(cube_coupling_map)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["To avoid that tedium though we could just use our SquareGrid Architecture:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.architecture import SquareGrid"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["alternative_cube_architecture = SquareGrid(2, 2, 2)\n", "draw_graph(alternative_cube_architecture.coupling)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The current range of quantum computers are commonly referred to as Noisy-Intermediate-Scale-Quantum devices i.e. NISQ devices. The impact of noise is a primary concern during compilation and incentivizes producing physically permitted circuits that have a minimal number of gates. For this reason benchmarking in this area is often completed by comparing the final number of two-qubit (or particularly SWAP gates) in compiled circuits."]}, {"cell_type": "markdown", "metadata": {}, "source": ["However it is important to remember that adding logical SWAP gates to minimise gate count is not the only way this constraint can be met, with large scale architecture-aware synthesis methods and fidelity aware methods amongst other approaches producing viable physically permitted circuits. It is likely that no SINGLE approach is better for all circuits, but the ability to use different approaches where best fitted will give the best results during compilation."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Producing physically valid circuits is completed via the `MappingManager` class, which aims to accomodate a wide range of approaches."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.mapping import MappingManager"]}, {"cell_type": "markdown", "metadata": {}, "source": ["A `MappingManager` object requires an `Architecture` object at construction."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["mapping_manager = MappingManager(id_architecture)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["All mapping is done through the `MappingManager.route_circuit` method. The `MappingManager.route_circuit` method has two arguments, the first a Circuit to be routed (which is mutated), the second a `List[RoutingMethodCircuit]` object that defines how the mapping is completed."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Later we will look at defining our own `RoutingMethodCircuit` objects, but initially lets consider one thats already available."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.mapping import LexiLabellingMethod, LexiRouteRoutingMethod"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["lexi_label = LexiLabellingMethod()\n", "lexi_route = LexiRouteRoutingMethod(10)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The `lexi_route` object here is of little use outside `MappingManager`. Note that it takes a lookahead parameter, which will affect the performance of the method, defining the number of two-qubit gates it considers when finding `SWAP` gates to add."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit, OpType\n", "from pytket.circuit import display"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = (\n", " Circuit(4)\n", " .CX(0, 1)\n", " .CX(1, 2)\n", " .CX(0, 2)\n", " .CX(0, 3)\n", " .CX(2, 3)\n", " .CX(1, 3)\n", " .CX(0, 1)\n", " .measure_all()\n", ")\n", "display.render_circuit_jupyter(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can also look at which logical qubits are interacting."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils import Graph"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["Graph(c).get_qubit_graph()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["By running the `MappingManager.route_circuit` method on our circuit `c` with the `LexiLabellingMethod` and `LexiRouteRoutingMethod` objects as an argument, qubits in `c` with some physical requirements will be relabelled and the qubit graph modified (by the addition of SWAP gates and relabelling some CX as BRIDGE gates) such that the qubit graph is isomorphic to some subgraph of the full architecture."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["mapping_manager.route_circuit(c, [lexi_label, lexi_route])\n", "display.render_circuit_jupyter(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The graph:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["Graph(c).get_qubit_graph()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The resulting circuit may also change if we reduce the lookahead parameter."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = (\n", " Circuit(4)\n", " .CX(0, 1)\n", " .CX(1, 2)\n", " .CX(0, 2)\n", " .CX(0, 3)\n", " .CX(2, 3)\n", " .CX(1, 3)\n", " .CX(0, 1)\n", " .measure_all()\n", ")\n", "mapping_manager.route_circuit(c, [lexi_label, LexiRouteRoutingMethod(1)])\n", "display.render_circuit_jupyter(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can also pass multiple `RoutingMethod` options for Routing in a ranked List. Each `RoutingMethod` option has a function for checking whether it can usefully modify a subcircuit at a stage in Routing. To choose, each method in the List is checked in order until one returns True. This will be discussed more later."]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can aid the mapping procedure by relabelling qubits in advance. This can be completed using the `Placement` class."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.placement import Placement, LinePlacement, GraphPlacement"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The default ```Placement``` assigns logical qubits to physical qubits as they are encountered during routing. ```LinePlacement``` uses a strategy described in https://arxiv.org/abs/1902.08091. ```GraphPlacement``` is described in Section 7.1 of https://arxiv.org/abs/2003.10611. Lets look at how we can use the ```LinePlacement``` class.`"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["line_placement = LinePlacement(id_architecture)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = (\n", " Circuit(4)\n", " .CX(0, 1)\n", " .CX(1, 2)\n", " .CX(0, 2)\n", " .CX(0, 3)\n", " .CX(2, 3)\n", " .CX(1, 3)\n", " .CX(0, 1)\n", " .measure_all()\n", ")\n", "line_placement.place(c)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["display.render_circuit_jupyter(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Note that one qubit remains unplaced in this example. `LexiRouteRoutingMethod` will dynamically assign it during mapping."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Different placements will lead to different selections of SWAP gates being added. However each different routed circuit will preserve the original unitary action of the full circuit while respecting connectivity constraints."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["mapping_manager.route_circuit(c, [lexi_label, lexi_route])\n", "display.render_circuit_jupyter(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The graph:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["Graph(c).get_qubit_graph()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["However, small changes to the depth of lookahead or the original assignment of `Architecture` `Node` can greatly affect the resulting physical circuit for the `LexiRouteRoutingMethod` method. Considering this variance, it should be possible to easily throw additional computational resources at the problem if necessary, which is something TKET is leaning towards with the ability to define custom `RoutingCircuitMethod` objects."]}, {"cell_type": "markdown", "metadata": {}, "source": ["To define a new `RoutingMethodCircuit` method though, we first need to understand how it is used in `MappingManager` and routing. The `MappingManager.route_circuit` method treats the global problem of mapping to physical circuits as many sequential sub-problems. Consider the following problem."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.placement import place_with_map"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ = Circuit(4).CX(0, 1).CX(1, 2).CX(0, 2).CX(0, 3).CX(2, 3).CX(1, 3).CX(0, 1)\n", "naive_map = {\n", " circ.qubits[0]: node_0,\n", " circ.qubits[1]: node_1,\n", " circ.qubits[2]: node_2,\n", " circ.qubits[3]: node_3,\n", "}\n", "place_with_map(circ, naive_map)\n", "Graph(circ).get_DAG()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["So what happens when we run the following?"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["mapping_manager.route_circuit(circ, [lexi_route])\n", "Graph(circ).get_DAG()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Sequential mapping typically works by partitioning the circuit into two, a first partition comprising a connected subcircuit that is physically permitted, a second partition that is not. Therefore, the first thing `MappingManager.route_circuit` does is find this partition for the passed circuit, by iterating through gates in the circuit."]}, {"cell_type": "markdown", "metadata": {}, "source": ["We will construct the partitions ourselves for illustrative purposes. Lets assume we are routing for the four qubit line architecture (qubits are connected to adjacent indices) \"simple_architecture\" we constructed earlier."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ_first_partition = Circuit(4).CX(0, 1).CX(1, 2)\n", "place_with_map(circ_first_partition, naive_map)\n", "Graph(circ_first_partition).get_DAG()"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ_second_partition = Circuit(4).CX(0, 2).CX(0, 3).CX(2, 3).CX(1, 3).CX(0, 1)\n", "place_with_map(circ_second_partition, naive_map)\n", "Graph(circ_second_partition).get_DAG()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Note that there are gates in the second partition that would be physically permitted, if they were not dependent on other gates that are not."]}, {"cell_type": "markdown", "metadata": {}, "source": ["The next step is to modify the second partition circuit to move it closer being physically permitted. Here the `LexiRouteRoutingMethod` as before will either insert a SWAP gate at the start of the partition, or will substitute a CX gate in the first slice of the partition with a BRIDGE gate."]}, {"cell_type": "markdown", "metadata": {}, "source": ["The option taken by `LexiRouteRoutingethod(1)` is to insert a SWAP gate between the first two nodes of the architecture, swapping their logical states. How does this change the second partition circuit?"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ_second_partition = (\n", " Circuit(4).SWAP(0, 1).CX(1, 2).CX(1, 3).CX(2, 3).CX(0, 3).CX(1, 0)\n", ")\n", "place_with_map(circ_second_partition, naive_map)\n", "Graph(circ_second_partition).get_DAG()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Leaving the full circuit as:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["full_circuit = (\n", " Circuit(4).CX(0, 1).CX(1, 2).SWAP(0, 1).CX(1, 2).CX(1, 3).CX(2, 3).CX(0, 3).CX(1, 0)\n", ")\n", "place_with_map(full_circuit, naive_map)\n", "Graph(full_circuit).get_DAG()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["After a modification is made the partition is updated."]}, {"cell_type": "markdown", "metadata": {}, "source": ["The first partition:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ_first_partition = Circuit(4).CX(0, 1).CX(1, 2).SWAP(0, 1).CX(1, 2)\n", "place_with_map(circ_first_partition, naive_map)\n", "Graph(circ_first_partition).get_DAG()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The second partition:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["circ_second_partition = Circuit(4).CX(1, 3).CX(2, 3).CX(0, 3).CX(1, 0)\n", "place_with_map(circ_second_partition, naive_map)\n", "Graph(circ_second_partition).get_DAG()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This pattern of modification and upating the partition is repeated until the partition has reached the end of the circuit, i.e. the back side of the partition has no gates in it. Also note that the process of updating the partition has been simplified for this example with \"physically permitted\" encapsulating two-qubit gate constraints only - in the future we expect other arity gates to provide constraints that need to be met. Also note that any modification to the second circuit can willfully modify the qubit labelling and a token swapping network will be automatically added to conform to the new labelling."]}, {"cell_type": "markdown", "metadata": {}, "source": ["We now enough about how `MappingManager` works to add our own `RoutingMethodCircuit`. While `LexiRouteRoutingMethod` is implemented in c++ TKET, giving it some advantages, via lambda functions we can define our own `RoutingMethodCircuit` in python."]}, {"cell_type": "markdown", "metadata": {}, "source": ["A python defined `RoutingMethodCircuit` requires three arguments. The first is a function that given a Circuit (the circuit after the partition) and an Architecture, returns a bool (determining whether the new circuit should be substitued in a full routing process), a new Circuit (a modification of the original circuit such as an added SWAP) a Dict between qubits reflecting any relabelling done in the method, and a Dict between qubits giving any implicit permutation of qubits (such as by adding a SWAP). For some clarity (we will write an example later), lets look at an example function declaration."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from typing import Dict"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def route_subcircuit_func(\n", " circuit: Circuit, architecture: Architecture\n", ") -> Tuple[bool, Circuit, Dict[Node, Node], Dict[Node, Node]]:\n", " return ()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The first return is a bool which detemrines if a given `RoutingMethodCircuit` is suitable for providing a solution at a given partition. `MappingManager.route_circuit` accepts a List of of `RoutingMethod` defining how solutions are found. At the point the partition circuit is modified, the circuit is passed to `RoutingMethodCircuit.routing_method` which additionally to finding a subcircuit substitution, should determine whether it can or can't helpfully modify the partition boundary circuit, and return True if it can. The first `RoutingMethodCircuit` to return True is then used for modification - meaning the ordering of List elements is important."]}, {"cell_type": "markdown", "metadata": {}, "source": ["The third argument sets the maximum number of gates given in the passed Circuit and the fourth argument sets the maximum depth in the passed Circuit."]}, {"cell_type": "markdown", "metadata": {}, "source": ["`LexiRouteRoutingMethod` will always return True, because it can always find some helpful SWAP to insert, and it can dynamically assign logical to physical qubits. Given this, lets construct a more specialised modification - an architecture-aware decomposition of a distance-2 CRy gate. Lets write our function type declarations for each method:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def distance2_CRy_decomp(\n", " circuit: Circuit, architecture: Architecture\n", ") -> Tuple[bool, Circuit, Dict[Node, Node], Dict[Node, Node]]:\n", " return (False, Circuit(), {}, {})"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Where do we start? Lets define a simple scope for our solution: for a single gate in the passed circuit (the circuit after the partition) that has OpType CRy, if the two qubits it's acting on are at distance 2 on the architecture, decompose the gate using BRIDGE gates."]}, {"cell_type": "markdown", "metadata": {}, "source": ["The first restriction is to only have a single gate from the first slice - we can achieve this by setting both the maximum depth and size parameters to 1."]}, {"cell_type": "markdown", "metadata": {}, "source": ["The second restriction is for the gate to have OpType CRy and for the qubits to be at distance 2 - we can check this restriction in a `distance2_CRy_check` method."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def distance2_CRy_check(circuit: Circuit, architecture: Architecture) -> bool:\n", " if circuit.n_gates != 1:\n", " raise ValueError(\n", " \"Circuit for CRy check should only have 1 gate, please change parameters of method declaration.\"\n", " )\n", " command = circuit.get_commands()[0]\n", " if command.op.type == OpType.CRy:\n", " # Architecture stores qubits under `Node` identifier\n", " n0 = Node(command.qubits[0].reg_name, command.qubits[0].index)\n", " n1 = Node(command.qubits[1].reg_name, command.qubits[1].index)\n", " # qubits could not be placed in circuit, so check before finding distance\n", " if n0 in architecture.nodes and n1 in architecture.nodes:\n", " # means we can run the decomposition\n", " if architecture.get_distance(n0, n1) == 2:\n", " return True\n", " return False"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The `distance2_CRy_check` confirms whether the required restrictions are respected. Given this, if the `distance2_CRy_decomp` method is called we know where to add the decomposition."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def distance2_CRy_decomp(\n", " circuit: Circuit, architecture: Architecture\n", ") -> Tuple[bool, Circuit, Dict[Node, Node], Dict[Node, Node]]:\n", " worthwhile_substitution = distance2_CRy_check(circuit, architecture)\n", " if worthwhile_substitution == False:\n", " return (False, Circuit(), {}, {})\n", " command = circuit.get_commands()[0]\n", " qubits = command.qubits\n", " # Architecture stores qubits under `Node` identifier\n", " n0 = Node(qubits[0].reg_name, qubits[0].index)\n", " n1 = Node(qubits[1].reg_name, qubits[1].index)\n\n", " # need to find connecting node for decomposition\n", " adjacent_nodes_0 = architecture.get_adjacent_nodes(n0)\n", " adjacent_nodes_1 = architecture.get_adjacent_nodes(n1)\n", " connecting_nodes = adjacent_nodes_0.intersection(adjacent_nodes_1)\n", " if len(connecting_nodes) == 0:\n", " raise ValueError(\"Qubits for distance-2 CRy decomp are not at distance 2.\")\n", " connecting_node = connecting_nodes.pop()\n", " c = Circuit()\n\n", " # the \"relabelling map\" empty, and the permutation map is qubit to qubit, so add here\n", " permutation_map = dict()\n", " for q in circuit.qubits:\n", " permutation_map[q] = q\n", " c.add_qubit(q)\n", " # rotation, can assume only parameter as CRy\n", " angle = command.op.params[0]\n", " c.Ry(angle, qubits[1])\n", " # distance-2 CX decomp\n", " c.CX(qubits[0], connecting_node).CX(connecting_node, qubits[1])\n", " c.CX(qubits[0], connecting_node).CX(connecting_node, qubits[1])\n", " # rotation\n", " c.Ry(-1 * angle, qubits[1])\n", " # distance-2 CX decomp\n", " c.CX(qubits[0], connecting_node).CX(connecting_node, qubits[1])\n", " c.CX(qubits[0], connecting_node).CX(connecting_node, qubits[1])\n\n", " # the \"relabelling map\" is just qubit to qubit\n", " return (True, c, {}, permutation_map)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Before turning this into a `RoutingMethod` we can try it ourselves."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_c = Circuit(4)\n", "test_c.CRy(0.6, 0, 2)\n", "place_with_map(test_c, naive_map)\n", "Graph(test_c).get_DAG()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["As we can see, our circuit has one CRy gate at distance two away."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(distance2_CRy_check(test_c, id_architecture))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Our method returns True, as expected! We should also test cases where it returns errors or False."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_c_false = Circuit(4)\n", "test_c_false.CRy(0.4, 0, 1)\n", "place_with_map(test_c_false, naive_map)\n", "print(distance2_CRy_check(test_c_false, id_architecture))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_c_error = Circuit(4)\n", "test_c_error.CRy(0.6, 0, 2)\n", "test_c_error.CRy(0.4, 0, 1)\n", "place_with_map(test_c_error, naive_map)\n", "try:\n", " distance2_CRy_check(test_c_error, id_architecture)\n", "except ValueError:\n", " print(\"Error reached!\")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Does the decomposition work?"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_c = Circuit(4)\n", "test_c.CRy(0.6, 0, 2)\n", "place_with_map(test_c, naive_map)\n", "decomp = distance2_CRy_decomp(test_c, id_architecture)\n", "display.render_circuit_jupyter(decomp[1])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Great! Our check function and decomposition method are both working. Lets wrap them into a `RoutingMethodCircuit` and try them out."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.mapping import RoutingMethodCircuit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["cry_rmc = RoutingMethodCircuit(distance2_CRy_decomp, 1, 1)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can use our original `MappingManager` object as it is defined for the same architecture. Lets try it out on a range of circumstances."]}, {"cell_type": "markdown", "metadata": {}, "source": ["If we pass it a full CX circuit without `LexiRouteRoutingMethod`, we should find that `MappingManager` throws an error, as none of the passed methods can route for the given circuit."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = (\n", " Circuit(4)\n", " .CX(0, 1)\n", " .CX(1, 2)\n", " .CX(0, 2)\n", " .CX(0, 3)\n", " .CX(2, 3)\n", " .CX(1, 3)\n", " .CX(0, 1)\n", " .measure_all()\n", ")\n", "place_with_map(c, naive_map)\n", "try:\n", " mapping_manager.route_circuit(c, [cry_rmc])\n", "except RuntimeError:\n", " print(\"Error reached!\")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Alternatively, we can add `LexiRouteRoutingMethod` on top:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = (\n", " Circuit(4)\n", " .CX(0, 1)\n", " .CX(1, 2)\n", " .CX(0, 2)\n", " .CX(0, 3)\n", " .CX(2, 3)\n", " .CX(1, 3)\n", " .CX(0, 1)\n", " .measure_all()\n", ")\n", "place_with_map(c, naive_map)\n", "mapping_manager.route_circuit(c, [cry_rmc, LexiRouteRoutingMethod(10)])\n", "display.render_circuit_jupyter(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["However as there are no CRy gates our new method is unused. We can add one:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = (\n", " Circuit(4)\n", " .CRy(0.6, 0, 2)\n", " .CX(0, 1)\n", " .CX(1, 2)\n", " .CX(0, 2)\n", " .CX(0, 3)\n", " .CX(2, 3)\n", " .CX(1, 3)\n", " .CX(0, 1)\n", " .measure_all()\n", ")\n", "mapping_manager.route_circuit(c, [lexi_label, cry_rmc, LexiRouteRoutingMethod(10)])\n", "display.render_circuit_jupyter(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This time we can see our decomposition! If we reorder the methods though `LexiRouteRoutingMethod` is checked first (and returns True), so our new method is unused. The order is important!"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Finally, lets see what happens if the gate is not at the right distance initially."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = (\n", " Circuit(4)\n", " .CRy(0.6, 0, 3)\n", " .CX(0, 1)\n", " .CX(1, 2)\n", " .CX(0, 2)\n", " .CX(0, 3)\n", " .CX(2, 3)\n", " .CX(1, 3)\n", " .CX(0, 1)\n", " .measure_all()\n", ")\n", "mapping_manager.route_circuit(c, [lexi_label, cry_rmc, LexiRouteRoutingMethod(10)])\n", "display.render_circuit_jupyter(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Above a SWAP gate is inserted by `LexiRouteRoutingMethod` before anything else."]}, {"cell_type": "markdown", "metadata": {}, "source": ["For anyone interested, a simple extension exercise could be to extend this to additionally work for distance-2 CRx and CRz. Alternatively one could improve on the method itself - this approach always decomposes a CRy at distance-2, but is this a good idea?"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Also note that higher performance solutions are coded straight into the TKET c++ codebase. This provides advantages, including that Circuit construction and substitution is unncessary (as with python) as the circuit can be directly modified, however the ability to produce prototypes at the python level is very helpful. If you have a great python implementation but are finding some runtime bottlenecks, why not try implementing it straight into TKET (the code is open source at https://github.com/CQCL/tket)."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Besides the `LexiRouteRoutingMethod()` and the `LexiLabellingMethod()` there are other routing methods in pytket, such as the `AASRouteRoutingMethod()` and the corresponding `AASLabellingMethod()`, which are used to route phase-polynomial boxes using architecture-aware synthesis. Usually circuits contain non-phase-polynomial operations as well, so it is a good idea to combine them with the `LexiRouteRoutingMethod()`, as in the following example:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.mapping import AASRouteRoutingMethod, AASLabellingMethod\n", "from pytket.circuit import PhasePolyBox, Qubit\n", "import numpy as np"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(3, 3)\n", "n_qb = 3\n", "qubit_indices = {Qubit(0): 0, Qubit(1): 1, Qubit(2): 2}\n", "phase_polynomial = {(True, False, True): 0.333, (False, False, True): 0.05}\n", "linear_transformation = np.array([[1, 0, 0], [0, 1, 0], [0, 0, 1]])\n", "p_box = PhasePolyBox(n_qb, qubit_indices, phase_polynomial, linear_transformation)\n", "c.add_phasepolybox(p_box, [0, 1, 2])\n", "c.CX(0, 1).CX(0, 2).CX(1, 2)\n", "display.render_circuit_jupyter(c)\n", "nodes = [Node(\"test\", 0), Node(\"test\", 1), Node(\"test\", 2)]\n", "arch = Architecture([[nodes[0], nodes[1]], [nodes[1], nodes[2]]])\n", "mm = MappingManager(arch)\n", "mm.route_circuit(\n", " c,\n", " [\n", " AASRouteRoutingMethod(1),\n", " LexiLabellingMethod(),\n", " LexiRouteRoutingMethod(),\n", " AASLabellingMethod(),\n", " ],\n", ")\n", "display.render_circuit_jupyter(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In this case the order of the methods is not very relevant, because in each step of the routing only one of the methods is suitable. In the first part of the circuit the mapping is done without inserting swaps by the AAS method; in the second part one swap gate is added to the circuit."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file diff --git a/examples/python/Forest_portability_example.py b/examples/python/Forest_portability_example.py index f834ad6b..af8def7f 100644 --- a/examples/python/Forest_portability_example.py +++ b/examples/python/Forest_portability_example.py @@ -1,4 +1,4 @@ -# # Code Portability and Intro to Forest +# # Code portability and intro to forest # The quantum hardware landscape is incredibly competitive and rapidly changing. Many full-stack quantum software platforms lock users into them in order to use the associated devices and simulators. This notebook demonstrates how `pytket` can free up your existing high-level code to be used on devices from other providers. We will take a state-preparation and evolution circuit generated using `qiskit`, and enable it to be run on several Rigetti backends. # diff --git a/examples/python/backends_example.py b/examples/python/backends_example.py index fb493e89..4343a272 100644 --- a/examples/python/backends_example.py +++ b/examples/python/backends_example.py @@ -1,4 +1,4 @@ -# # TKET Backend Tutorial +# # TKET backend tutorial # This example shows how to use `pytket` to execute quantum circuits on both simulators and real devices, and how to interpret the results. As tket is designed to be platform-agnostic, we have unified the interfaces of different providers as much as possible into the `Backend` class for maximum portability of code. The following is a selection of currently supported backends: # * ProjectQ simulator diff --git a/examples/python/conditional_gate_example.py b/examples/python/conditional_gate_example.py index 81bb8110..dad54ebf 100644 --- a/examples/python/conditional_gate_example.py +++ b/examples/python/conditional_gate_example.py @@ -1,4 +1,4 @@ -# # Conditional Gates +# # Conditional gates # Whilst any quantum process can be created by performing "pure" operations delaying all measurements to the end, this is not always practical and can greatly increase the resource requirements. It is much more convenient to alternate quantum gates and measurements, especially if we can use the measurement results to determine which gates to apply (we refer to this more generic circuit model as "mixed" circuits, against the usual "pure" circuits). This is especially crucial for error correcting codes, where the correction gates are applied only if an error is detected. # diff --git a/examples/python/entanglement_swapping.py b/examples/python/entanglement_swapping.py index 1c39beb1..2d75d330 100644 --- a/examples/python/entanglement_swapping.py +++ b/examples/python/entanglement_swapping.py @@ -1,4 +1,4 @@ -# # Iterated Entanglement Swapping using TKET +# # Iterated entanglement swapping using TKET # In this tutorial, we will focus on: # - designing circuits with mid-circuit measurement and conditional gates; diff --git a/examples/python/expectation_value_example.py b/examples/python/expectation_value_example.py index ad7ac9dc..6c5aa573 100644 --- a/examples/python/expectation_value_example.py +++ b/examples/python/expectation_value_example.py @@ -1,4 +1,4 @@ -# # Expectation Values +# # Expectation values # Given a circuit generating a quantum state $\lvert \psi \rangle$, it is very common to have an operator $H$ and ask for the expectation value $\langle \psi \vert H \vert \psi \rangle$. A notable example is in quantum computational chemistry, where $\lvert \psi \rangle$ encodes the wavefunction for the electronic state of a small molecule, and the energy of the molecule can be derived from the expectation value with respect to the molecule's Hamiltonian operator $H$. # diff --git a/examples/python/mapping_example.py b/examples/python/mapping_example.py index 9de6c6fb..7f82fb1e 100644 --- a/examples/python/mapping_example.py +++ b/examples/python/mapping_example.py @@ -1,4 +1,4 @@ -# # Qubit Mapping and Routing +# # Qubit mapping and routing # In this tutorial we will show how the problem of mapping from logical quantum circuits to physically permitted circuits is solved automatically in TKET. The basic examples require only the installation of pytket, ```pip install pytket```. diff --git a/examples/python/pytket-qujax-classification.py b/examples/python/pytket-qujax-classification.py index 44b07a00..a8214a0c 100644 --- a/examples/python/pytket-qujax-classification.py +++ b/examples/python/pytket-qujax-classification.py @@ -1,4 +1,4 @@ -# # Binary Classification using pytket-qujax +# # Binary classification using pytket-qujax from jax import numpy as jnp, random, vmap, value_and_grad, jit from pytket import Circuit diff --git a/examples/pytket-qujax-classification.ipynb b/examples/pytket-qujax-classification.ipynb index e55499f2..115720dc 100644 --- a/examples/pytket-qujax-classification.ipynb +++ b/examples/pytket-qujax-classification.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Binary Classification using pytket-qujax"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from jax import numpy as jnp, random, vmap, value_and_grad, jit\n", "from pytket import Circuit\n", "from pytket.circuit.display import render_circuit_jupyter\n", "from pytket.extensions.qujax.qujax_convert import tk_to_qujax\n", "import matplotlib.pyplot as plt"]}, {"cell_type": "markdown", "metadata": {}, "source": ["# Define the classification task
\n", "We'll try and learn a _donut_ binary classification function (i.e. a bivariate coordinate is labelled 1 if it is inside the donut and 0 if it is outside)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["inner_rad = 0.25\n", "outer_rad = 0.75"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def classification_function(x, y):\n", " r = jnp.sqrt(x**2 + y**2)\n", " return jnp.where((r > inner_rad) * (r < outer_rad), 1, 0)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["linsp = jnp.linspace(-1, 1, 1000)\n", "Z = vmap(lambda x: vmap(lambda y: classification_function(x, y))(linsp))(linsp)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["plt.contourf(linsp, linsp, Z, cmap=\"Purples\")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now let's generate some data for our quantum circuit to learn from"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["n_data = 1000\n", "x = random.uniform(random.PRNGKey(0), shape=(n_data, 2), minval=-1, maxval=1)\n", "y = classification_function(x[:, 0], x[:, 1])"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["plt.scatter(x[:, 0], x[:, 1], alpha=jnp.where(y, 1, 0.2), s=10)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["# Quantum circuit time
\n", "We'll use a variant of data re-uploading [P\u00e9rez-Salinas et al](https://doi.org/10.22331/q-2020-02-06-226) to encode the input data, alongside some variational parameters within a quantum circuit classifier"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["n_qubits = 3\n", "depth = 5"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(n_qubits)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["for layer in range(depth):\n", " for qi in range(n_qubits):\n", " c.Rz(0.0, qi)\n", " c.Ry(0.0, qi)\n", " c.Rz(0.0, qi)\n", " if layer < (depth - 1):\n", " for qi in range(layer, layer + n_qubits - 1, 2):\n", " c.CZ(qi % n_qubits, (qi + 1) % n_qubits)\n", " c.add_barrier(range(n_qubits))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["render_circuit_jupyter(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can use `pytket-qujax` to generate our angles-to-statetensor function."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["angles_to_st = tk_to_qujax(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We'll parameterise each angle as
\n", "$$ \\theta_k = b_k + w_k * x_k $$
\n", "where $b_k, w_k$ are variational parameters to be learnt and $x_k = x_0$ if $k$ even, $x_k = x_1$ if $k$ odd for a single bivariate input point $(x_0, x_1)$."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["n_angles = 3 * n_qubits * depth\n", "n_params = 2 * n_angles"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def param_and_x_to_angles(param, x_single):\n", " biases = param[:n_angles]\n", " weights = param[n_angles:]\n", " weights_times_data = jnp.where(\n", " jnp.arange(n_angles) % 2 == 0, weights * x_single[0], weights * x_single[1]\n", " )\n", " angles = biases + weights_times_data\n", " return angles"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["param_and_x_to_st = lambda param, x_single: angles_to_st(\n", " param_and_x_to_angles(param, x_single)\n", ")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We'll measure the first qubit only (if its 1 we label _donut_, if its 0 we label _not donut_)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def param_and_x_to_probability(param, x_single):\n", " st = param_and_x_to_st(param, x_single)\n", " all_probs = jnp.square(jnp.abs(st))\n", " first_qubit_probs = jnp.sum(all_probs, axis=range(1, n_qubits))\n", " return first_qubit_probs[1]"]}, {"cell_type": "markdown", "metadata": {}, "source": ["For binary classification, the likelihood for our full data set $(x_{1:N}, y_{1:N})$ is
\n", "$$ p(y_{1:N} \\mid b, w, x_{1:N}) = \\prod_{i=1}^N p(y_i \\mid b, w, x_i) = \\prod_{i=1}^N (1 - q_{(b,w)}(x_i))^{\\mathbb{I}[y_i = 0]}q_{(b,w)}(x_i)^{\\mathbb{I}[y_i = 1]}, $$
\n", "where $q_{(b, w)}(x)$ is the probability the quantum circuit classifies input $x$ as donut given variational parameter vectors $(b, w)$. This gives log-likelihood
\n", "$$ \\log p(y_{1:N} \\mid b, w, x_{1:N}) = \\sum_{i=1}^N \\mathbb{I}[y_i = 0] \\log(1 - q_{(b,w)}(x_i)) + \\mathbb{I}[y_i = 1] \\log q_{(b,w)}(x_i), $$
\n", "which we would like to maximise.
\n", "
\n", "Unfortunately, the log-likelihood **cannot** be approximated unbiasedly using shots, that is we can approximate $q_{(b,w)}(x_i)$ unbiasedly but not $\\log(q_{(b,w)}(x_i))$.
\n", "Note that in qujax simulations we can use the statetensor to calculate this exactly, but it is still good to keep in mind loss functions that can also be used with shots from a quantum device."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Instead we can minimise an expected distance between shots and data
\n", "

\n", "$$ C(b, w, x, y) = \\mathbb{E}_{p(y' \\mid q_{(b, w)}(x))}[\\ell(y', y)] = (1 - q_{(b, w)}(x)) \\ell(0, y) + q_{(b, w)}(x)\\ell(1, y), $$
\n", "

\n", "where $y'$ is a shot, $y$ is a data label and $\\ell$ is some distance between bitstrings - here we simply set $\\ell(0, 0) = \\ell(1, 1) = 0$ and $\\ell(0, 1) = \\ell(1, 0) = 1$ (which coincides with the Hamming distance for this binary example). The full batch cost function is $C(b, w) = \\frac1N \\sum_{i=1}^N C(b, w, x_i, y_i)$.
\n", "
\n", "Note that to calculate the cost function we need to evaluate the statetensor for every input point $x_i$. If the dataset becomes too large, we can easily minibatch."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def param_to_cost(param):\n", " donut_probs = vmap(param_and_x_to_probability, in_axes=(None, 0))(param, x)\n", " costs = jnp.where(y, 1 - donut_probs, donut_probs)\n", " return costs.mean()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["# Ready to descend some gradients?
\n", "We'll just use vanilla gradient descent here"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["param_to_cost_and_grad = jit(value_and_grad(param_to_cost))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["n_iter = 1000\n", "stepsize = 1e-1\n", "param = random.uniform(random.PRNGKey(1), shape=(n_params,), minval=0, maxval=2)\n", "costs = jnp.zeros(n_iter)\n", "for i in range(n_iter):\n", " cost, grad = param_to_cost_and_grad(param)\n", " costs = costs.at[i].set(cost)\n", " param = param - stepsize * grad\n", " print(i, \"Cost: \", cost, end=\"\\r\")"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["plt.plot(costs)\n", "plt.xlabel(\"Iteration\")\n", "plt.ylabel(\"Cost\")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["# Visualise trained classifier"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["linsp = jnp.linspace(-1, 1, 100)\n", "Z = vmap(\n", " lambda a: vmap(lambda b: param_and_x_to_probability(param, jnp.array([a, b])))(\n", " linsp\n", " )\n", ")(linsp)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["plt.contourf(linsp, linsp, Z, cmap=\"Purples\", alpha=0.8)\n", "circle_linsp = jnp.linspace(0, 2 * jnp.pi, 100)\n", "plt.plot(inner_rad * jnp.cos(circle_linsp), inner_rad * jnp.sin(circle_linsp), c=\"red\")\n", "plt.plot(outer_rad * jnp.cos(circle_linsp), outer_rad * jnp.sin(circle_linsp), c=\"red\")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Looks good, it has clearly grasped the donut shape. Sincerest apologies if you are now hungry! \ud83c\udf69"]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Binary classification using pytket-qujax"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from jax import numpy as jnp, random, vmap, value_and_grad, jit\n", "from pytket import Circuit\n", "from pytket.circuit.display import render_circuit_jupyter\n", "from pytket.extensions.qujax.qujax_convert import tk_to_qujax\n", "import matplotlib.pyplot as plt"]}, {"cell_type": "markdown", "metadata": {}, "source": ["# Define the classification task
\n", "We'll try and learn a _donut_ binary classification function (i.e. a bivariate coordinate is labelled 1 if it is inside the donut and 0 if it is outside)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["inner_rad = 0.25\n", "outer_rad = 0.75"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def classification_function(x, y):\n", " r = jnp.sqrt(x**2 + y**2)\n", " return jnp.where((r > inner_rad) * (r < outer_rad), 1, 0)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["linsp = jnp.linspace(-1, 1, 1000)\n", "Z = vmap(lambda x: vmap(lambda y: classification_function(x, y))(linsp))(linsp)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["plt.contourf(linsp, linsp, Z, cmap=\"Purples\")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now let's generate some data for our quantum circuit to learn from"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["n_data = 1000\n", "x = random.uniform(random.PRNGKey(0), shape=(n_data, 2), minval=-1, maxval=1)\n", "y = classification_function(x[:, 0], x[:, 1])"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["plt.scatter(x[:, 0], x[:, 1], alpha=jnp.where(y, 1, 0.2), s=10)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["# Quantum circuit time
\n", "We'll use a variant of data re-uploading [P\u00e9rez-Salinas et al](https://doi.org/10.22331/q-2020-02-06-226) to encode the input data, alongside some variational parameters within a quantum circuit classifier"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["n_qubits = 3\n", "depth = 5"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(n_qubits)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["for layer in range(depth):\n", " for qi in range(n_qubits):\n", " c.Rz(0.0, qi)\n", " c.Ry(0.0, qi)\n", " c.Rz(0.0, qi)\n", " if layer < (depth - 1):\n", " for qi in range(layer, layer + n_qubits - 1, 2):\n", " c.CZ(qi % n_qubits, (qi + 1) % n_qubits)\n", " c.add_barrier(range(n_qubits))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["render_circuit_jupyter(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can use `pytket-qujax` to generate our angles-to-statetensor function."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["angles_to_st = tk_to_qujax(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We'll parameterise each angle as
\n", "$$ \\theta_k = b_k + w_k * x_k $$
\n", "where $b_k, w_k$ are variational parameters to be learnt and $x_k = x_0$ if $k$ even, $x_k = x_1$ if $k$ odd for a single bivariate input point $(x_0, x_1)$."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["n_angles = 3 * n_qubits * depth\n", "n_params = 2 * n_angles"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def param_and_x_to_angles(param, x_single):\n", " biases = param[:n_angles]\n", " weights = param[n_angles:]\n", " weights_times_data = jnp.where(\n", " jnp.arange(n_angles) % 2 == 0, weights * x_single[0], weights * x_single[1]\n", " )\n", " angles = biases + weights_times_data\n", " return angles"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["param_and_x_to_st = lambda param, x_single: angles_to_st(\n", " param_and_x_to_angles(param, x_single)\n", ")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We'll measure the first qubit only (if its 1 we label _donut_, if its 0 we label _not donut_)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def param_and_x_to_probability(param, x_single):\n", " st = param_and_x_to_st(param, x_single)\n", " all_probs = jnp.square(jnp.abs(st))\n", " first_qubit_probs = jnp.sum(all_probs, axis=range(1, n_qubits))\n", " return first_qubit_probs[1]"]}, {"cell_type": "markdown", "metadata": {}, "source": ["For binary classification, the likelihood for our full data set $(x_{1:N}, y_{1:N})$ is
\n", "$$ p(y_{1:N} \\mid b, w, x_{1:N}) = \\prod_{i=1}^N p(y_i \\mid b, w, x_i) = \\prod_{i=1}^N (1 - q_{(b,w)}(x_i))^{\\mathbb{I}[y_i = 0]}q_{(b,w)}(x_i)^{\\mathbb{I}[y_i = 1]}, $$
\n", "where $q_{(b, w)}(x)$ is the probability the quantum circuit classifies input $x$ as donut given variational parameter vectors $(b, w)$. This gives log-likelihood
\n", "$$ \\log p(y_{1:N} \\mid b, w, x_{1:N}) = \\sum_{i=1}^N \\mathbb{I}[y_i = 0] \\log(1 - q_{(b,w)}(x_i)) + \\mathbb{I}[y_i = 1] \\log q_{(b,w)}(x_i), $$
\n", "which we would like to maximise.
\n", "
\n", "Unfortunately, the log-likelihood **cannot** be approximated unbiasedly using shots, that is we can approximate $q_{(b,w)}(x_i)$ unbiasedly but not $\\log(q_{(b,w)}(x_i))$.
\n", "Note that in qujax simulations we can use the statetensor to calculate this exactly, but it is still good to keep in mind loss functions that can also be used with shots from a quantum device."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Instead we can minimise an expected distance between shots and data
\n", "

\n", "$$ C(b, w, x, y) = \\mathbb{E}_{p(y' \\mid q_{(b, w)}(x))}[\\ell(y', y)] = (1 - q_{(b, w)}(x)) \\ell(0, y) + q_{(b, w)}(x)\\ell(1, y), $$
\n", "

\n", "where $y'$ is a shot, $y$ is a data label and $\\ell$ is some distance between bitstrings - here we simply set $\\ell(0, 0) = \\ell(1, 1) = 0$ and $\\ell(0, 1) = \\ell(1, 0) = 1$ (which coincides with the Hamming distance for this binary example). The full batch cost function is $C(b, w) = \\frac1N \\sum_{i=1}^N C(b, w, x_i, y_i)$.
\n", "
\n", "Note that to calculate the cost function we need to evaluate the statetensor for every input point $x_i$. If the dataset becomes too large, we can easily minibatch."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def param_to_cost(param):\n", " donut_probs = vmap(param_and_x_to_probability, in_axes=(None, 0))(param, x)\n", " costs = jnp.where(y, 1 - donut_probs, donut_probs)\n", " return costs.mean()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["# Ready to descend some gradients?
\n", "We'll just use vanilla gradient descent here"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["param_to_cost_and_grad = jit(value_and_grad(param_to_cost))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["n_iter = 1000\n", "stepsize = 1e-1\n", "param = random.uniform(random.PRNGKey(1), shape=(n_params,), minval=0, maxval=2)\n", "costs = jnp.zeros(n_iter)\n", "for i in range(n_iter):\n", " cost, grad = param_to_cost_and_grad(param)\n", " costs = costs.at[i].set(cost)\n", " param = param - stepsize * grad\n", " print(i, \"Cost: \", cost, end=\"\\r\")"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["plt.plot(costs)\n", "plt.xlabel(\"Iteration\")\n", "plt.ylabel(\"Cost\")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["# Visualise trained classifier"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["linsp = jnp.linspace(-1, 1, 100)\n", "Z = vmap(\n", " lambda a: vmap(lambda b: param_and_x_to_probability(param, jnp.array([a, b])))(\n", " linsp\n", " )\n", ")(linsp)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["plt.contourf(linsp, linsp, Z, cmap=\"Purples\", alpha=0.8)\n", "circle_linsp = jnp.linspace(0, 2 * jnp.pi, 100)\n", "plt.plot(inner_rad * jnp.cos(circle_linsp), inner_rad * jnp.sin(circle_linsp), c=\"red\")\n", "plt.plot(outer_rad * jnp.cos(circle_linsp), outer_rad * jnp.sin(circle_linsp), c=\"red\")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Looks good, it has clearly grasped the donut shape. Sincerest apologies if you are now hungry! \ud83c\udf69"]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file From 3974481046629ab920c51b59dbc85bbff0d987cc Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 26 Oct 2023 18:02:36 +0100 Subject: [PATCH 19/51] TOC edit --- examples/_toc.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/examples/_toc.yml b/examples/_toc.yml index c5c32481..692002ff 100644 --- a/examples/_toc.yml +++ b/examples/_toc.yml @@ -32,8 +32,8 @@ parts: - file: pytket-qujax_heisenberg_vqe - caption: Other chapters: - - file: tket_benchmarking - file: benchmarking/README + - file: tket_benchmarking - file: entanglement_swapping - file: spam_example - file: expectation_value_example From d286ad43e6891a3632f1c503630c13e02f30658c Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 26 Oct 2023 18:12:42 +0100 Subject: [PATCH 20/51] modify some more headings --- examples/ansatz_sequence_example.ipynb | 2 +- examples/measurement_reduction_example.ipynb | 2 +- examples/python/ansatz_sequence_example.py | 2 +- examples/python/measurement_reduction_example.py | 2 +- examples/python/ucc_vqe.py | 2 +- examples/ucc_vqe.ipynb | 2 +- 6 files changed, 6 insertions(+), 6 deletions(-) diff --git a/examples/ansatz_sequence_example.ipynb b/examples/ansatz_sequence_example.ipynb index db714284..1ef80ee4 100644 --- a/examples/ansatz_sequence_example.ipynb +++ b/examples/ansatz_sequence_example.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Ansatz Sequencing"]}, {"cell_type": "markdown", "metadata": {}, "source": ["When performing variational algorithms like VQE, one common approach to generating circuit ans\u00e4tze is to take an operator $U$ representing excitations and use this to act on a reference state $\\lvert \\phi_0 \\rangle$. One such ansatz is the Unitary Coupled Cluster ansatz. Each excitation, indexed by $j$, within $U$ is given a real coefficient $a_j$ and a parameter $t_j$, such that $U = e^{i \\sum_j \\sum_k a_j t_j P_{jk}}$, where $P_{jk} \\in \\{I, X, Y, Z \\}^{\\otimes n}$. The exact form is dependent on the chosen qubit encoding. This excitation gives us a variational state $\\lvert \\psi (t) \\rangle = U(t) \\lvert \\phi_0 \\rangle$. The operator $U$ must be Trotterised, to give a product of Pauli exponentials, and converted into native quantum gates to create the ansatz circuit.
\n", "
\n", "This notebook will describe how to use an advanced feature of `pytket` to enable automated circuit synthesis for $U$ and reduce circuit depth dramatically.
\n", "
\n", "We must create a `pytket` `QubitPauliOperator`, which represents such an operator $U$, and contains a dictionary from Pauli string $P_{jk}$ to symbolic expression. Here, we make a mock operator ourselves, which resembles the UCCSD excitation operator for the $\\mathrm{H}_2$ molecule using the Jordan-Wigner qubit encoding. In the future, operator generation will be handled automatically using CQC's upcoming software for enterprise quantum chemistry, EUMEN. We also offer conversion to and from the `OpenFermion` `QubitOperator` class, although at the time of writing a `QubitOperator` cannot handle arbitrary symbols.
\n", "
\n", "First, we create a series of `QubitPauliString` objects, which represent each $P_{jk}$."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.circuit import Qubit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["q = [Qubit(i) for i in range(4)]\n", "qps0 = QubitPauliString([q[0], q[1], q[2]], [Pauli.Y, Pauli.Z, Pauli.X])\n", "qps1 = QubitPauliString([q[0], q[1], q[2]], [Pauli.X, Pauli.Z, Pauli.Y])\n", "qps2 = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.X, Pauli.Y])\n", "qps3 = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.Y, Pauli.X])\n", "qps4 = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.X, Pauli.X])\n", "qps5 = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.X, Pauli.X])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now, create some symbolic expressions for the $a_j t_j$ terms."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import fresh_symbol"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["symbol1 = fresh_symbol(\"s0\")\n", "expr1 = 1.2 * symbol1\n", "symbol2 = fresh_symbol(\"s1\")\n", "expr2 = -0.3 * symbol2"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can now create our `QubitPauliOperator`."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils import QubitPauliOperator"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["dict1 = dict((string, expr1) for string in (qps0, qps1))\n", "dict2 = dict((string, expr2) for string in (qps2, qps3, qps4, qps5))\n", "operator = QubitPauliOperator({**dict1, **dict2})\n", "print(operator)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now we can let `pytket` sequence the terms in this operator for us, using a selection of strategies. First, we will create a `Circuit` to generate an example reference state, and then use the `gen_term_sequence_circuit` method to append the Pauli exponentials."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import Circuit\n", "from pytket.utils import gen_term_sequence_circuit\n", "from pytket.partition import PauliPartitionStrat, GraphColourMethod"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["reference_circ = Circuit(4).X(1).X(3)\n", "ansatz_circuit = gen_term_sequence_circuit(\n", " operator, reference_circ, PauliPartitionStrat.CommutingSets, GraphColourMethod.Lazy\n", ")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This method works by generating a graph of Pauli exponentials and performing graph colouring. Here we have chosen to partition the terms so that exponentials which commute are gathered together, and we have done so using a lazy, greedy graph colouring method.
\n", "
\n", "Alternatively, we could have used the `PauliPartitionStrat.NonConflictingSets`, which puts Pauli exponentials together so that they only require single-qubit gates to be converted into the form $e^{i \\alpha Z \\otimes Z \\otimes ... \\otimes Z}$. This strategy is primarily useful for measurement reduction, a different problem.
\n", "
\n", "We could also have used the `GraphColourMethod.LargestFirst`, which still uses a greedy method, but builds the full graph and iterates through the vertices in descending order of arity. We recommend playing around with the options, but we typically find that the combination of `CommutingSets` and `Lazy` allows the best optimisation.
\n", "
\n", "In general, not all of our exponentials will commute, so the semantics of our circuit depend on the order of our sequencing. As a result, it is important for us to be able to inspect the order we have produced. `pytket` provides functionality to enable this. Each set of commuting exponentials is put into a `CircBox`, which lets us inspect the partitoning."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import OpType"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["for command in ansatz_circuit:\n", " if command.op.type == OpType.CircBox:\n", " print(\"New CircBox:\")\n", " for pauli_exp in command.op.get_circuit():\n", " print(\n", " \" {} {} {}\".format(\n", " pauli_exp, pauli_exp.op.get_paulis(), pauli_exp.op.get_phase()\n", " )\n", " )\n", " else:\n", " print(\"Native gate: {}\".format(command))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can convert this circuit into basic gates using a `pytket` `Transform`. This acts in place on the circuit to do rewriting, for gate translation and optimisation. We will start off with a naive decomposition."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.transform import Transform"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["naive_circuit = ansatz_circuit.copy()\n", "Transform.DecomposeBoxes().apply(naive_circuit)\n", "print(naive_circuit.get_commands())"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is a jumble of one- and two-qubit gates. We can get some relevant circuit metrics out:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"Naive CX Depth: {}\".format(naive_circuit.depth_by_type(OpType.CX)))\n", "print(\"Naive CX Count: {}\".format(naive_circuit.n_gates_of_type(OpType.CX)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These metrics can be improved upon significantly by smart compilation. A `Transform` exists precisely for this purpose:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.transform import PauliSynthStrat, CXConfigType"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["smart_circuit = ansatz_circuit.copy()\n", "Transform.UCCSynthesis(PauliSynthStrat.Sets, CXConfigType.Tree).apply(smart_circuit)\n", "print(\"Smart CX Depth: {}\".format(smart_circuit.depth_by_type(OpType.CX)))\n", "print(\"Smart CX Count: {}\".format(smart_circuit.n_gates_of_type(OpType.CX)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This `Transform` takes in a `Circuit` with the structure specified above: some arbitrary gates for the reference state, along with several `CircBox` gates containing `PauliExpBox` gates.
\n", "
\n", "We have chosen `PauliSynthStrat.Sets` and `CXConfigType.Tree`. The `PauliSynthStrat` dictates the method for decomposing multiple adjacent Pauli exponentials into basic gates, while the `CXConfigType` dictates the structure of adjacent CX gates.
\n", "
\n", "If we choose a different combination of strategies, we can produce a different output circuit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["last_circuit = ansatz_circuit.copy()\n", "Transform.UCCSynthesis(PauliSynthStrat.Individual, CXConfigType.Snake).apply(\n", " last_circuit\n", ")\n", "print(last_circuit.get_commands())"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"Last CX Depth: {}\".format(last_circuit.depth_by_type(OpType.CX)))\n", "print(\"Last CX Count: {}\".format(last_circuit.n_gates_of_type(OpType.CX)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Other than some single-qubit Cliffords we acquired via synthesis, you can check that this gives us the same circuit structure as our `Transform.DecomposeBoxes` method! It is a suboptimal synthesis method.
\n", "
\n", "As with the `gen_term_sequence` method, we recommend playing around with the arguments and seeing what circuits come out. Typically we find that `PauliSynthStrat.Sets` and `CXConfigType.Tree` work the best, although routing can affect this somewhat."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Ansatz sequencing"]}, {"cell_type": "markdown", "metadata": {}, "source": ["When performing variational algorithms like VQE, one common approach to generating circuit ans\u00e4tze is to take an operator $U$ representing excitations and use this to act on a reference state $\\lvert \\phi_0 \\rangle$. One such ansatz is the Unitary Coupled Cluster ansatz. Each excitation, indexed by $j$, within $U$ is given a real coefficient $a_j$ and a parameter $t_j$, such that $U = e^{i \\sum_j \\sum_k a_j t_j P_{jk}}$, where $P_{jk} \\in \\{I, X, Y, Z \\}^{\\otimes n}$. The exact form is dependent on the chosen qubit encoding. This excitation gives us a variational state $\\lvert \\psi (t) \\rangle = U(t) \\lvert \\phi_0 \\rangle$. The operator $U$ must be Trotterised, to give a product of Pauli exponentials, and converted into native quantum gates to create the ansatz circuit.
\n", "
\n", "This notebook will describe how to use an advanced feature of `pytket` to enable automated circuit synthesis for $U$ and reduce circuit depth dramatically.
\n", "
\n", "We must create a `pytket` `QubitPauliOperator`, which represents such an operator $U$, and contains a dictionary from Pauli string $P_{jk}$ to symbolic expression. Here, we make a mock operator ourselves, which resembles the UCCSD excitation operator for the $\\mathrm{H}_2$ molecule using the Jordan-Wigner qubit encoding. In the future, operator generation will be handled automatically using CQC's upcoming software for enterprise quantum chemistry, EUMEN. We also offer conversion to and from the `OpenFermion` `QubitOperator` class, although at the time of writing a `QubitOperator` cannot handle arbitrary symbols.
\n", "
\n", "First, we create a series of `QubitPauliString` objects, which represent each $P_{jk}$."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.circuit import Qubit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["q = [Qubit(i) for i in range(4)]\n", "qps0 = QubitPauliString([q[0], q[1], q[2]], [Pauli.Y, Pauli.Z, Pauli.X])\n", "qps1 = QubitPauliString([q[0], q[1], q[2]], [Pauli.X, Pauli.Z, Pauli.Y])\n", "qps2 = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.X, Pauli.Y])\n", "qps3 = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.Y, Pauli.X])\n", "qps4 = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.X, Pauli.X])\n", "qps5 = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.X, Pauli.X])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now, create some symbolic expressions for the $a_j t_j$ terms."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import fresh_symbol"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["symbol1 = fresh_symbol(\"s0\")\n", "expr1 = 1.2 * symbol1\n", "symbol2 = fresh_symbol(\"s1\")\n", "expr2 = -0.3 * symbol2"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can now create our `QubitPauliOperator`."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils import QubitPauliOperator"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["dict1 = dict((string, expr1) for string in (qps0, qps1))\n", "dict2 = dict((string, expr2) for string in (qps2, qps3, qps4, qps5))\n", "operator = QubitPauliOperator({**dict1, **dict2})\n", "print(operator)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now we can let `pytket` sequence the terms in this operator for us, using a selection of strategies. First, we will create a `Circuit` to generate an example reference state, and then use the `gen_term_sequence_circuit` method to append the Pauli exponentials."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import Circuit\n", "from pytket.utils import gen_term_sequence_circuit\n", "from pytket.partition import PauliPartitionStrat, GraphColourMethod"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["reference_circ = Circuit(4).X(1).X(3)\n", "ansatz_circuit = gen_term_sequence_circuit(\n", " operator, reference_circ, PauliPartitionStrat.CommutingSets, GraphColourMethod.Lazy\n", ")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This method works by generating a graph of Pauli exponentials and performing graph colouring. Here we have chosen to partition the terms so that exponentials which commute are gathered together, and we have done so using a lazy, greedy graph colouring method.
\n", "
\n", "Alternatively, we could have used the `PauliPartitionStrat.NonConflictingSets`, which puts Pauli exponentials together so that they only require single-qubit gates to be converted into the form $e^{i \\alpha Z \\otimes Z \\otimes ... \\otimes Z}$. This strategy is primarily useful for measurement reduction, a different problem.
\n", "
\n", "We could also have used the `GraphColourMethod.LargestFirst`, which still uses a greedy method, but builds the full graph and iterates through the vertices in descending order of arity. We recommend playing around with the options, but we typically find that the combination of `CommutingSets` and `Lazy` allows the best optimisation.
\n", "
\n", "In general, not all of our exponentials will commute, so the semantics of our circuit depend on the order of our sequencing. As a result, it is important for us to be able to inspect the order we have produced. `pytket` provides functionality to enable this. Each set of commuting exponentials is put into a `CircBox`, which lets us inspect the partitoning."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import OpType"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["for command in ansatz_circuit:\n", " if command.op.type == OpType.CircBox:\n", " print(\"New CircBox:\")\n", " for pauli_exp in command.op.get_circuit():\n", " print(\n", " \" {} {} {}\".format(\n", " pauli_exp, pauli_exp.op.get_paulis(), pauli_exp.op.get_phase()\n", " )\n", " )\n", " else:\n", " print(\"Native gate: {}\".format(command))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can convert this circuit into basic gates using a `pytket` `Transform`. This acts in place on the circuit to do rewriting, for gate translation and optimisation. We will start off with a naive decomposition."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.transform import Transform"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["naive_circuit = ansatz_circuit.copy()\n", "Transform.DecomposeBoxes().apply(naive_circuit)\n", "print(naive_circuit.get_commands())"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is a jumble of one- and two-qubit gates. We can get some relevant circuit metrics out:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"Naive CX Depth: {}\".format(naive_circuit.depth_by_type(OpType.CX)))\n", "print(\"Naive CX Count: {}\".format(naive_circuit.n_gates_of_type(OpType.CX)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These metrics can be improved upon significantly by smart compilation. A `Transform` exists precisely for this purpose:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.transform import PauliSynthStrat, CXConfigType"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["smart_circuit = ansatz_circuit.copy()\n", "Transform.UCCSynthesis(PauliSynthStrat.Sets, CXConfigType.Tree).apply(smart_circuit)\n", "print(\"Smart CX Depth: {}\".format(smart_circuit.depth_by_type(OpType.CX)))\n", "print(\"Smart CX Count: {}\".format(smart_circuit.n_gates_of_type(OpType.CX)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This `Transform` takes in a `Circuit` with the structure specified above: some arbitrary gates for the reference state, along with several `CircBox` gates containing `PauliExpBox` gates.
\n", "
\n", "We have chosen `PauliSynthStrat.Sets` and `CXConfigType.Tree`. The `PauliSynthStrat` dictates the method for decomposing multiple adjacent Pauli exponentials into basic gates, while the `CXConfigType` dictates the structure of adjacent CX gates.
\n", "
\n", "If we choose a different combination of strategies, we can produce a different output circuit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["last_circuit = ansatz_circuit.copy()\n", "Transform.UCCSynthesis(PauliSynthStrat.Individual, CXConfigType.Snake).apply(\n", " last_circuit\n", ")\n", "print(last_circuit.get_commands())"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"Last CX Depth: {}\".format(last_circuit.depth_by_type(OpType.CX)))\n", "print(\"Last CX Count: {}\".format(last_circuit.n_gates_of_type(OpType.CX)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Other than some single-qubit Cliffords we acquired via synthesis, you can check that this gives us the same circuit structure as our `Transform.DecomposeBoxes` method! It is a suboptimal synthesis method.
\n", "
\n", "As with the `gen_term_sequence` method, we recommend playing around with the arguments and seeing what circuits come out. Typically we find that `PauliSynthStrat.Sets` and `CXConfigType.Tree` work the best, although routing can affect this somewhat."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file diff --git a/examples/measurement_reduction_example.ipynb b/examples/measurement_reduction_example.ipynb index daaf6b2f..5b6ea777 100644 --- a/examples/measurement_reduction_example.ipynb +++ b/examples/measurement_reduction_example.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Advanced Expectation Values and Measurement Reduction"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This notebook is an advanced follow-up to the \"expectation_value_example\" notebook, focussing on reducing the number of circuits required for measurement.
\n", "
\n", "When calculating the expectation value $\\langle \\psi \\vert H \\vert \\psi \\rangle$ of some operator $H$ on a quantum computer, we prepare $\\vert \\psi \\rangle$ using a circuit, and the operator $H$ is first decomposed into a sum of smaller, tractable operators of the form $\\alpha P$, where $P \\in \\mathcal{G}_n$, the multi-qubit Pauli group. Naively, one would obtain the expectation value of each of these smaller operators individually by doing shots on the quantum computer and measuring in the correct Pauli bases. Assuming the device measures only single qubits in the $Z$-basis, this basis change requires single-qubit Clifford gates, which are \"cheaper\" (less noisy and quicker) than entangling gates. The sum of these smaller operator expectation values is then used to obtain the desired $\\langle \\psi \\vert H \\vert \\psi \\rangle$.
\n", "
\n", "However, the scaling of this process can be poor, meaning that many shots are required. Instead, several of these smaller operators can be measured simultaneously, reducing the total number of measurements. For some sets of measurements, it can be done \"for free\", meaning that no extra entangling gates are required to perform simultaneous measurement. For general commuting sets of Pauli measurements, Clifford gates are required for simultaneous measurement, including entangling gates."]}, {"cell_type": "markdown", "metadata": {}, "source": ["There are several strategies for measurement reduction throughout the literature. Examples include https://arxiv.org/abs/1908.06942, https://arxiv.org/abs/1908.08067 and https://arxiv.org/abs/1907.07859."]}, {"cell_type": "markdown", "metadata": {}, "source": ["In `pytket`, we provide tools to perform measurement reduction. The most accessible way is to use the utils method, `get_operator_expectation_value`. This method wraps up some under-the-hood processes to allow users to calculate expectation values, agnostic to the backend, operator, or circuit. In this tutorial we will use the Qiskit Aer simulators via the `AerBackend`, for shots, and the `AerStateBackend`, for statevector simulation.
\n", "
\n", "We use the `QubitPauliOperator` class to represent the operator $H$."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import Circuit, Qubit\n", "from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.utils import QubitPauliOperator\n", "from pytket.utils.expectations import get_operator_expectation_value\n", "from pytket.extensions.qiskit import AerBackend, AerStateBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["First, let's get some results on a toy circuit without using any measurement reduction:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["shots_backend = AerBackend()\n", "n_shots = 10000"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(5)\n", "c.H(4)\n", "c.V(2)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = shots_backend.get_compiled_circuit(c)\n", "op = QubitPauliOperator(\n", " {\n", " QubitPauliString([Qubit(0)], [Pauli.Z]): 0.1,\n", " QubitPauliString(\n", " [Qubit(0), Qubit(1), Qubit(2), Qubit(3), Qubit(4)],\n", " [Pauli.Y, Pauli.Z, Pauli.X, Pauli.X, Pauli.Y],\n", " ): 0.4,\n", " QubitPauliString([Qubit(0), Qubit(1)], [Pauli.X, Pauli.X]): 0.2,\n", " }\n", ")"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["shots_result = get_operator_expectation_value(c, op, shots_backend, n_shots)\n", "print(shots_result)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The result should be around 0.1, although as the shot simulator is stochastic this will be inexact. Let's test to check what the exact result should be using the statevector simulator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["state_backend = AerStateBackend()\n", "state_result = get_operator_expectation_value(c, op, state_backend)\n", "print(state_result)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now we can introduce measurement reduction. First we need to choose a strategy:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.partition import PauliPartitionStrat"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This first one only performs measurements on simultaneous Pauli operators when there is no cost incurred to do so."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["strat = PauliPartitionStrat.NonConflictingSets\n", "shots_result = get_operator_expectation_value(c, op, shots_backend, n_shots, strat)\n", "print(shots_result)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The other strategy we use groups together arbitrary Pauli operators, with the condition that all Pauli operators within a group commute. For an input circuit with $n$ qubits, our method requires the addition of up to $\\frac{n(n-1)}{2}$ $CX$ gates to \"diagonalise\" the Pauli operators, although in practice we find that our techniques tend to give far lower gate overhead than this bound. We describe the procedure in an upcoming paper."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["strat = PauliPartitionStrat.CommutingSets\n", "shots_result = get_operator_expectation_value(c, op, shots_backend, n_shots, strat)\n", "print(shots_result)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Obviously, the `AerBackend` can be swapped out for the backend of a real machine."]}, {"cell_type": "markdown", "metadata": {}, "source": ["We will now demonstrate how to manually use the methods that are being called by `get_operator_expectation_value`. These methods are primarily intended for internal use, but we show them here for advanced users who may wish to have more information about the number of CX gates being added to each circuit, the number of circuits being run and other diagnostics."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import OpType\n", "from pytket.partition import measurement_reduction"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["id_string = QubitPauliString()\n", "qpt_list = [p for p in op._dict.keys() if (p != id_string)]\n", "setup_1 = measurement_reduction(qpt_list, PauliPartitionStrat.NonConflictingSets)\n", "print(\"Circuits required for measurement: {}\".format(len(setup_1.measurement_circs)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This produced a `MeasurementSetup` object using the `NonConflictingSets` strategy of measurement reduction. This object holds a set of circuits which perform different basis changes, and the measurements associated with these circuits.
\n", "
\n", "There are 3 circuits held within the `MeasurementSetup` object, meaning that our original `QubitOperator` has been reduced from the 5 originally required measurements to 3."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["for circ in setup_1.measurement_circs:\n", " print(\"CX gates for measurement: {}\".format(circ.n_gates_of_type(OpType.CX)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["No CX gates have been added for any of the required measurements. Now, we will change to the `CommutingSets` strategy."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["setup_2 = measurement_reduction(qpt_list, PauliPartitionStrat.CommutingSets)\n", "print(\"Circuits required for measurement: {}\".format(len(setup_2.measurement_circs)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["There are only 2 circuits required when expanding the scope of allowed simultaneous measurements. However, this comes at a cost:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["for circ in setup_2.measurement_circs:\n", " print(\"CX gates for measurement: {}\".format(circ.n_gates_of_type(OpType.CX)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["A CX gate has been introduced to one of the measurement circuits, to convert to the correct Pauli basis set. On current devices which are extremely constrained in the number of entangling gates, the reduction in number of shots may not be worth the gate overhead."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Advanced expectation values and measurement reduction"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This notebook is an advanced follow-up to the \"expectation_value_example\" notebook, focussing on reducing the number of circuits required for measurement.
\n", "
\n", "When calculating the expectation value $\\langle \\psi \\vert H \\vert \\psi \\rangle$ of some operator $H$ on a quantum computer, we prepare $\\vert \\psi \\rangle$ using a circuit, and the operator $H$ is first decomposed into a sum of smaller, tractable operators of the form $\\alpha P$, where $P \\in \\mathcal{G}_n$, the multi-qubit Pauli group. Naively, one would obtain the expectation value of each of these smaller operators individually by doing shots on the quantum computer and measuring in the correct Pauli bases. Assuming the device measures only single qubits in the $Z$-basis, this basis change requires single-qubit Clifford gates, which are \"cheaper\" (less noisy and quicker) than entangling gates. The sum of these smaller operator expectation values is then used to obtain the desired $\\langle \\psi \\vert H \\vert \\psi \\rangle$.
\n", "
\n", "However, the scaling of this process can be poor, meaning that many shots are required. Instead, several of these smaller operators can be measured simultaneously, reducing the total number of measurements. For some sets of measurements, it can be done \"for free\", meaning that no extra entangling gates are required to perform simultaneous measurement. For general commuting sets of Pauli measurements, Clifford gates are required for simultaneous measurement, including entangling gates."]}, {"cell_type": "markdown", "metadata": {}, "source": ["There are several strategies for measurement reduction throughout the literature. Examples include https://arxiv.org/abs/1908.06942, https://arxiv.org/abs/1908.08067 and https://arxiv.org/abs/1907.07859."]}, {"cell_type": "markdown", "metadata": {}, "source": ["In `pytket`, we provide tools to perform measurement reduction. The most accessible way is to use the utils method, `get_operator_expectation_value`. This method wraps up some under-the-hood processes to allow users to calculate expectation values, agnostic to the backend, operator, or circuit. In this tutorial we will use the Qiskit Aer simulators via the `AerBackend`, for shots, and the `AerStateBackend`, for statevector simulation.
\n", "
\n", "We use the `QubitPauliOperator` class to represent the operator $H$."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import Circuit, Qubit\n", "from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.utils import QubitPauliOperator\n", "from pytket.utils.expectations import get_operator_expectation_value\n", "from pytket.extensions.qiskit import AerBackend, AerStateBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["First, let's get some results on a toy circuit without using any measurement reduction:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["shots_backend = AerBackend()\n", "n_shots = 10000"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(5)\n", "c.H(4)\n", "c.V(2)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = shots_backend.get_compiled_circuit(c)\n", "op = QubitPauliOperator(\n", " {\n", " QubitPauliString([Qubit(0)], [Pauli.Z]): 0.1,\n", " QubitPauliString(\n", " [Qubit(0), Qubit(1), Qubit(2), Qubit(3), Qubit(4)],\n", " [Pauli.Y, Pauli.Z, Pauli.X, Pauli.X, Pauli.Y],\n", " ): 0.4,\n", " QubitPauliString([Qubit(0), Qubit(1)], [Pauli.X, Pauli.X]): 0.2,\n", " }\n", ")"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["shots_result = get_operator_expectation_value(c, op, shots_backend, n_shots)\n", "print(shots_result)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The result should be around 0.1, although as the shot simulator is stochastic this will be inexact. Let's test to check what the exact result should be using the statevector simulator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["state_backend = AerStateBackend()\n", "state_result = get_operator_expectation_value(c, op, state_backend)\n", "print(state_result)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now we can introduce measurement reduction. First we need to choose a strategy:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.partition import PauliPartitionStrat"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This first one only performs measurements on simultaneous Pauli operators when there is no cost incurred to do so."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["strat = PauliPartitionStrat.NonConflictingSets\n", "shots_result = get_operator_expectation_value(c, op, shots_backend, n_shots, strat)\n", "print(shots_result)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The other strategy we use groups together arbitrary Pauli operators, with the condition that all Pauli operators within a group commute. For an input circuit with $n$ qubits, our method requires the addition of up to $\\frac{n(n-1)}{2}$ $CX$ gates to \"diagonalise\" the Pauli operators, although in practice we find that our techniques tend to give far lower gate overhead than this bound. We describe the procedure in an upcoming paper."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["strat = PauliPartitionStrat.CommutingSets\n", "shots_result = get_operator_expectation_value(c, op, shots_backend, n_shots, strat)\n", "print(shots_result)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Obviously, the `AerBackend` can be swapped out for the backend of a real machine."]}, {"cell_type": "markdown", "metadata": {}, "source": ["We will now demonstrate how to manually use the methods that are being called by `get_operator_expectation_value`. These methods are primarily intended for internal use, but we show them here for advanced users who may wish to have more information about the number of CX gates being added to each circuit, the number of circuits being run and other diagnostics."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import OpType\n", "from pytket.partition import measurement_reduction"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["id_string = QubitPauliString()\n", "qpt_list = [p for p in op._dict.keys() if (p != id_string)]\n", "setup_1 = measurement_reduction(qpt_list, PauliPartitionStrat.NonConflictingSets)\n", "print(\"Circuits required for measurement: {}\".format(len(setup_1.measurement_circs)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This produced a `MeasurementSetup` object using the `NonConflictingSets` strategy of measurement reduction. This object holds a set of circuits which perform different basis changes, and the measurements associated with these circuits.
\n", "
\n", "There are 3 circuits held within the `MeasurementSetup` object, meaning that our original `QubitOperator` has been reduced from the 5 originally required measurements to 3."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["for circ in setup_1.measurement_circs:\n", " print(\"CX gates for measurement: {}\".format(circ.n_gates_of_type(OpType.CX)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["No CX gates have been added for any of the required measurements. Now, we will change to the `CommutingSets` strategy."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["setup_2 = measurement_reduction(qpt_list, PauliPartitionStrat.CommutingSets)\n", "print(\"Circuits required for measurement: {}\".format(len(setup_2.measurement_circs)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["There are only 2 circuits required when expanding the scope of allowed simultaneous measurements. However, this comes at a cost:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["for circ in setup_2.measurement_circs:\n", " print(\"CX gates for measurement: {}\".format(circ.n_gates_of_type(OpType.CX)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["A CX gate has been introduced to one of the measurement circuits, to convert to the correct Pauli basis set. On current devices which are extremely constrained in the number of entangling gates, the reduction in number of shots may not be worth the gate overhead."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file diff --git a/examples/python/ansatz_sequence_example.py b/examples/python/ansatz_sequence_example.py index fcb7845c..35efcc50 100644 --- a/examples/python/ansatz_sequence_example.py +++ b/examples/python/ansatz_sequence_example.py @@ -1,4 +1,4 @@ -# # Ansatz Sequencing +# # Ansatz sequencing # When performing variational algorithms like VQE, one common approach to generating circuit ansätze is to take an operator $U$ representing excitations and use this to act on a reference state $\lvert \phi_0 \rangle$. One such ansatz is the Unitary Coupled Cluster ansatz. Each excitation, indexed by $j$, within $U$ is given a real coefficient $a_j$ and a parameter $t_j$, such that $U = e^{i \sum_j \sum_k a_j t_j P_{jk}}$, where $P_{jk} \in \{I, X, Y, Z \}^{\otimes n}$. The exact form is dependent on the chosen qubit encoding. This excitation gives us a variational state $\lvert \psi (t) \rangle = U(t) \lvert \phi_0 \rangle$. The operator $U$ must be Trotterised, to give a product of Pauli exponentials, and converted into native quantum gates to create the ansatz circuit. # diff --git a/examples/python/measurement_reduction_example.py b/examples/python/measurement_reduction_example.py index 7775b643..8385741c 100644 --- a/examples/python/measurement_reduction_example.py +++ b/examples/python/measurement_reduction_example.py @@ -1,4 +1,4 @@ -# # Advanced Expectation Values and Measurement Reduction +# # Advanced expectation values and measurement reduction # This notebook is an advanced follow-up to the "expectation_value_example" notebook, focussing on reducing the number of circuits required for measurement. # diff --git a/examples/python/ucc_vqe.py b/examples/python/ucc_vqe.py index d09500fc..bb529be9 100644 --- a/examples/python/ucc_vqe.py +++ b/examples/python/ucc_vqe.py @@ -1,4 +1,4 @@ -# # VQE for Unitary Coupled Cluster using TKET +# # VQE with UCC ansatz using TKET # In this tutorial, we will focus on: # - building parameterised ansätze for variational algorithms; diff --git a/examples/ucc_vqe.ipynb b/examples/ucc_vqe.ipynb index 5400c6c3..b976650e 100644 --- a/examples/ucc_vqe.ipynb +++ b/examples/ucc_vqe.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# VQE for Unitary Coupled Cluster using TKET"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In this tutorial, we will focus on:
\n", "- building parameterised ans\u00e4tze for variational algorithms;
\n", "- compilation tools for UCC-style ans\u00e4tze."]}, {"cell_type": "markdown", "metadata": {}, "source": ["This example assumes the reader is familiar with the Variational Quantum Eigensolver and its application to electronic structure problems through the Unitary Coupled Cluster approach.
\n", "
\n", "To run this example, you will need `pytket` and `pytket-qiskit`, as well as `openfermion`, `scipy`, and `sympy`.
\n", "
\n", "We will start with a basic implementation and then gradually modify it to make it faster, more general, and less noisy. The final solution is given in full at the bottom of the notebook.
\n", "
\n", "Suppose we have some electronic configuration problem, expressed via a physical Hamiltonian. (The Hamiltonian and excitations in this example were obtained using `qiskit-aqua` version 0.5.2 and `pyscf` for H2, bond length 0.75A, sto3g basis, Jordan-Wigner encoding, with no qubit reduction or orbital freezing.). We express it succinctly using the openfermion library:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import openfermion as of"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian = (\n", " -0.8153001706270075 * of.QubitOperator(\"\")\n", " + 0.16988452027940318 * of.QubitOperator(\"Z0\")\n", " + -0.21886306781219608 * of.QubitOperator(\"Z1\")\n", " + 0.16988452027940323 * of.QubitOperator(\"Z2\")\n", " + -0.2188630678121961 * of.QubitOperator(\"Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z0 Z1\")\n", " + 0.16821198673715723 * of.QubitOperator(\"Z0 Z2\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z0 Z3\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z1 Z2\")\n", " + 0.1739537877649417 * of.QubitOperator(\"Z1 Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z2 Z3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 Y2 Y3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 Y2 Y3\")\n", ")\n", "nuclear_repulsion_energy = 0.70556961456"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We would like to define our ansatz for arbitrary parameter values. For simplicity, let's start with a Hardware Efficient Ansatz."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Hardware efficient ansatz:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def hea(params):\n", " ansatz = Circuit(4)\n", " for i in range(4):\n", " ansatz.Ry(params[i], i)\n", " for i in range(3):\n", " ansatz.CX(i, i + 1)\n", " for i in range(4):\n", " ansatz.Ry(params[4 + i], i)\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can use this to build the objective function for our optimisation."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import AerBackend\n", "from pytket.utils.expectations import expectation_from_counts"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Naive objective function:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " energy = 0\n", " for term, coeff in hamiltonian.terms.items():\n", " if not term:\n", " energy += coeff\n", " continue\n", " circ = hea(params)\n", " circ.add_c_register(\"c\", len(term))\n", " for i, (q, pauli) in enumerate(term):\n", " if pauli == \"X\":\n", " circ.H(q)\n", " elif pauli == \"Y\":\n", " circ.V(q)\n", " circ.Measure(q, i)\n", " compiled_circ = backend.get_compiled_circuit(circ)\n", " counts = backend.run_circuit(compiled_circ, n_shots=4000).get_counts()\n", " energy += coeff * expectation_from_counts(counts)\n", " return energy + nuclear_repulsion_energy"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This objective function is then run through a classical optimiser to find the set of parameter values that minimise the energy of the system. For the sake of example, we will just run this with a single parameter value."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["arg_values = [\n", " -7.31158201e-02,\n", " -1.64514836e-04,\n", " 1.12585591e-03,\n", " -2.58367544e-03,\n", " 1.00006068e00,\n", " -1.19551357e-03,\n", " 9.99963988e-01,\n", " 2.53283285e-03,\n", "]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["energy = objective(arg_values)\n", "print(energy)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The HEA is designed to cram as many orthogonal degrees of freedom into a small circuit as possible to be able to explore a large region of the Hilbert space whilst the circuits themselves can be run with minimal noise. These ans\u00e4tze give virtually-optimal circuits by design, but suffer from an excessive number of variational parameters making convergence slow, barren plateaus where the classical optimiser fails to make progress, and spanning a space where most states lack a physical interpretation. These drawbacks can necessitate adding penalties and may mean that the ansatz cannot actually express the true ground state.
\n", "
\n", "The UCC ansatz, on the other hand, is derived from the electronic configuration. It sacrifices efficiency of the circuit for the guarantee of physical states and the variational parameters all having some meaningful effect, which helps the classical optimisation to converge.
\n", "
\n", "This starts by defining the terms of our single and double excitations. These would usually be generated using the orbital configurations, so we will just use a hard-coded example here for the purposes of demonstration."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import Qubit\n", "from pytket.pauli import Pauli, QubitPauliString"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["q = [Qubit(i) for i in range(4)]\n", "xyii = QubitPauliString([q[0], q[1]], [Pauli.X, Pauli.Y])\n", "yxii = QubitPauliString([q[0], q[1]], [Pauli.Y, Pauli.X])\n", "iixy = QubitPauliString([q[2], q[3]], [Pauli.X, Pauli.Y])\n", "iiyx = QubitPauliString([q[2], q[3]], [Pauli.Y, Pauli.X])\n", "xxxy = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.X, Pauli.Y])\n", "xxyx = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.Y, Pauli.X])\n", "xyxx = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.X, Pauli.X])\n", "yxxx = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.X, Pauli.X])\n", "yyyx = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.Y, Pauli.X])\n", "yyxy = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.X, Pauli.Y])\n", "yxyy = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.Y, Pauli.Y])\n", "xyyy = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.Y, Pauli.Y])"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["singles_a = {xyii: 1.0, yxii: -1.0}\n", "singles_b = {iixy: 1.0, iiyx: -1.0}\n", "doubles = {\n", " xxxy: 0.25,\n", " xxyx: -0.25,\n", " xyxx: 0.25,\n", " yxxx: -0.25,\n", " yyyx: -0.25,\n", " yyxy: 0.25,\n", " yxyy: -0.25,\n", " xyyy: 0.25,\n", "}"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Building the ansatz circuit itself is often done naively by defining the map from each term down to basic gates and then applying it to each term."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def add_operator_term(circuit: Circuit, term: QubitPauliString, angle: float):\n", " qubits = []\n", " for q, p in term.map.items():\n", " if p != Pauli.I:\n", " qubits.append(q)\n", " if p == Pauli.X:\n", " circuit.H(q)\n", " elif p == Pauli.Y:\n", " circuit.V(q)\n", " for i in range(len(qubits) - 1):\n", " circuit.CX(i, i + 1)\n", " circuit.Rz(angle, len(qubits) - 1)\n", " for i in reversed(range(len(qubits) - 1)):\n", " circuit.CX(i, i + 1)\n", " for q, p in term.map.items():\n", " if p == Pauli.X:\n", " circuit.H(q)\n", " elif p == Pauli.Y:\n", " circuit.Vdg(q)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Unitary Coupled Cluster Singles & Doubles ansatz:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def ucc(params):\n", " ansatz = Circuit(4)\n", " # Set initial reference state\n", " ansatz.X(1).X(3)\n", " # Evolve by excitations\n", " for term, coeff in singles_a.items():\n", " add_operator_term(ansatz, term, coeff * params[0])\n", " for term, coeff in singles_b.items():\n", " add_operator_term(ansatz, term, coeff * params[1])\n", " for term, coeff in doubles.items():\n", " add_operator_term(ansatz, term, coeff * params[2])\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is already quite verbose, but `pytket` has a neat shorthand construction for these operator terms using the `PauliExpBox` construction. We can then decompose these into basic gates using the `DecomposeBoxes` compiler pass."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import PauliExpBox\n", "from pytket.passes import DecomposeBoxes"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def add_excitation(circ, term_dict, param):\n", " for term, coeff in term_dict.items():\n", " qubits, paulis = zip(*term.map.items())\n", " pbox = PauliExpBox(paulis, coeff * param)\n", " circ.add_pauliexpbox(pbox, qubits)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["UCC ansatz with syntactic shortcuts:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def ucc(params):\n", " ansatz = Circuit(4)\n", " ansatz.X(1).X(3)\n", " add_excitation(ansatz, singles_a, params[0])\n", " add_excitation(ansatz, singles_b, params[1])\n", " add_excitation(ansatz, doubles, params[2])\n", " DecomposeBoxes().apply(ansatz)\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The objective function can also be simplified using a utility method for constructing the measurement circuits and processing for expectation value calculations. For that, we convert the Hamiltonian to a pytket QubitPauliOperator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["pauli_sym = {\"I\": Pauli.I, \"X\": Pauli.X, \"Y\": Pauli.Y, \"Z\": Pauli.Z}"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qps_from_openfermion(paulis):\n", " \"\"\"Convert OpenFermion tensor of Paulis to pytket QubitPauliString.\"\"\"\n", " qlist = []\n", " plist = []\n", " for q, p in paulis:\n", " qlist.append(Qubit(q))\n", " plist.append(pauli_sym[p])\n", " return QubitPauliString(qlist, plist)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qpo_from_openfermion(openf_op):\n", " \"\"\"Convert OpenFermion QubitOperator to pytket QubitPauliOperator.\"\"\"\n", " tk_op = dict()\n", " for term, coeff in openf_op.terms.items():\n", " string = qps_from_openfermion(term)\n", " tk_op[string] = coeff\n", " return QubitPauliOperator(tk_op)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian_op = qpo_from_openfermion(hamiltonian)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Simplified objective function using utilities:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils.expectations import get_operator_expectation_value"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc(params)\n", " return (\n", " get_operator_expectation_value(circ, hamiltonian_op, backend, n_shots=4000)\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["arg_values = [-3.79002933e-05, 2.42964799e-05, 4.63447157e-01]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["energy = objective(arg_values)\n", "print(energy)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is now the simplest form that this operation can take, but it isn't necessarily the most effective. When we decompose the ansatz circuit into basic gates, it is still very expensive. We can employ some of the circuit simplification passes available in `pytket` to reduce its size and improve fidelity in practice.
\n", "
\n", "A good example is to decompose each `PauliExpBox` into basic gates and then apply `FullPeepholeOptimise`, which defines a compilation strategy utilising all of the simplifications in `pytket` that act locally on small regions of a circuit. We can examine the effectiveness by looking at the number of two-qubit gates before and after simplification, which tends to be a good indicator of fidelity for near-term systems where these gates are often slow and inaccurate."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import OpType\n", "from pytket.passes import FullPeepholeOptimise"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_circuit = ucc(arg_values)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count before\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth before\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["FullPeepholeOptimise().apply(test_circuit)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count after FPO\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth after FPO\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These simplification techniques are very general and are almost always beneficial to apply to a circuit if you want to eliminate local redundancies. But UCC ans\u00e4tze have extra structure that we can exploit further. They are defined entirely out of exponentiated tensors of Pauli matrices, giving the regular structure described by the `PauliExpBox`es. Under many circumstances, it is more efficient to not synthesise these constructions individually, but simultaneously in groups. The `PauliSimp` pass finds the description of a given circuit as a sequence of `PauliExpBox`es and resynthesises them (by default, in groups of commuting terms). This can cause great change in the overall structure and shape of the circuit, enabling the identification and elimination of non-local redundancy."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.passes import PauliSimp"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_circuit = ucc(arg_values)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count before\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth before\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["PauliSimp().apply(test_circuit)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count after PS\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth after PS\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["FullPeepholeOptimise().apply(test_circuit)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count after PS+FPO\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth after PS+FPO\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["To include this into our routines, we can just add the simplification passes to the objective function. The `get_operator_expectation_value` utility handles compiling to meet the requirements of the backend, so we don't have to worry about that here."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function with circuit simplification:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc(params)\n", " PauliSimp().apply(circ)\n", " FullPeepholeOptimise().apply(circ)\n", " return (\n", " get_operator_expectation_value(circ, hamiltonian_op, backend, n_shots=4000)\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These circuit simplification techniques have tried to preserve the exact unitary of the circuit, but there are ways to change the unitary whilst preserving the correctness of the algorithm as a whole.
\n", "
\n", "For example, the excitation terms are generated by trotterisation of the excitation operator, and the order of the terms does not change the unitary in the limit of many trotter steps, so in this sense we are free to sequence the terms how we like and it is sensible to do this in a way that enables efficient synthesis of the circuit. Prioritising collecting terms into commuting sets is a very beneficial heuristic for this and can be performed using the `gen_term_sequence_circuit` method to group the terms together into collections of `PauliExpBox`es and the `GuidedPauliSimp` pass to utilise these sets for synthesis."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.passes import GuidedPauliSimp\n", "from pytket.utils import gen_term_sequence_circuit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def ucc(params):\n", " singles_params = {qps: params[0] * coeff for qps, coeff in singles.items()}\n", " doubles_params = {qps: params[1] * coeff for qps, coeff in doubles.items()}\n", " excitation_op = QubitPauliOperator({**singles_params, **doubles_params})\n", " reference_circ = Circuit(4).X(1).X(3)\n", " ansatz = gen_term_sequence_circuit(excitation_op, reference_circ)\n", " GuidedPauliSimp().apply(ansatz)\n", " FullPeepholeOptimise().apply(ansatz)\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Adding these simplification routines doesn't come for free. Compiling and simplifying the circuit to achieve the best results possible can be a difficult task, which can take some time for the classical computer to perform.
\n", "
\n", "During a VQE run, we will call this objective function many times and run many measurement circuits within each, but the circuits that are run on the quantum computer are almost identical, having the same gate structure but with different gate parameters and measurements. We have already exploited this within the body of the objective function by simplifying the ansatz circuit before we call `get_operator_expectation_value`, so it is only done once per objective calculation rather than once per measurement circuit.
\n", "
\n", "We can go even further by simplifying it once outside of the objective function, and then instantiating the simplified ansatz with the parameter values needed. For this, we will construct the UCC ansatz circuit using symbolic (parametric) gates."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from sympy import symbols"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Symbolic UCC ansatz generation:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["syms = symbols(\"p0 p1 p2\")\n", "singles_a_syms = {qps: syms[0] * coeff for qps, coeff in singles_a.items()}\n", "singles_b_syms = {qps: syms[1] * coeff for qps, coeff in singles_b.items()}\n", "doubles_syms = {qps: syms[2] * coeff for qps, coeff in doubles.items()}\n", "excitation_op = QubitPauliOperator({**singles_a_syms, **singles_b_syms, **doubles_syms})\n", "ucc_ref = Circuit(4).X(1).X(3)\n", "ucc = gen_term_sequence_circuit(excitation_op, ucc_ref)\n", "GuidedPauliSimp().apply(ucc)\n", "FullPeepholeOptimise().apply(ucc)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function using the symbolic ansatz:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc.copy()\n", " sym_map = dict(zip(syms, params))\n", " circ.symbol_substitution(sym_map)\n", " return (\n", " get_operator_expectation_value(circ, hamiltonian_op, backend, n_shots=4000)\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We have now got some very good use of `pytket` for simplifying each individual circuit used in our experiment and for minimising the amount of time spent compiling, but there is still more we can do in terms of reducing the amount of work the quantum computer has to do. Currently, each (non-trivial) term in our measurement hamiltonian is measured by a different circuit within each expectation value calculation. Measurement reduction techniques exist for identifying when these observables commute and hence can be simultaneously measured, reducing the number of circuits required for the full expectation value calculation.
\n", "
\n", "This is built in to the `get_operator_expectation_value` method and can be applied by specifying a way to partition the measuremrnt terms. `PauliPartitionStrat.CommutingSets` can greatly reduce the number of measurement circuits by combining any number of terms that mutually commute. However, this involves potentially adding an arbitrary Clifford circuit to change the basis of the measurements which can be costly on NISQ devices, so `PauliPartitionStrat.NonConflictingSets` trades off some of the reduction in circuit number to guarantee that only single-qubit gates are introduced."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.partition import PauliPartitionStrat"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function using measurement reduction:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc.copy()\n", " sym_map = dict(zip(syms, params))\n", " circ.symbol_substitution(sym_map)\n", " return (\n", " get_operator_expectation_value(\n", " circ,\n", " operator,\n", " backend,\n", " n_shots=4000,\n", " partition_strat=PauliPartitionStrat.CommutingSets,\n", " )\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "markdown", "metadata": {}, "source": ["At this point, we have completely transformed how our VQE objective function works, improving its resilience to noise, cutting the number of circuits run, and maintaining fast runtimes. In doing this, we have explored a number of the features `pytket` offers that are beneficial to VQE and the UCC method:
\n", "- high-level syntactic constructs for evolution operators;
\n", "- utility methods for easy expectation value calculations;
\n", "- both generic and domain-specific circuit simplification methods;
\n", "- symbolic circuit compilation;
\n", "- measurement reduction for expectation value calculations."]}, {"cell_type": "markdown", "metadata": {}, "source": ["For the sake of completeness, the following gives the full code for the final solution, including passing the objective function to a classical optimiser to find the ground state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import openfermion as of\n", "from scipy.optimize import minimize\n", "from sympy import symbols"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import AerBackend\n", "from pytket.circuit import Circuit, Qubit\n", "from pytket.partition import PauliPartitionStrat\n", "from pytket.passes import GuidedPauliSimp, FullPeepholeOptimise\n", "from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.utils import get_operator_expectation_value, gen_term_sequence_circuit\n", "from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Obtain electronic Hamiltonian:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian = (\n", " -0.8153001706270075 * of.QubitOperator(\"\")\n", " + 0.16988452027940318 * of.QubitOperator(\"Z0\")\n", " + -0.21886306781219608 * of.QubitOperator(\"Z1\")\n", " + 0.16988452027940323 * of.QubitOperator(\"Z2\")\n", " + -0.2188630678121961 * of.QubitOperator(\"Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z0 Z1\")\n", " + 0.16821198673715723 * of.QubitOperator(\"Z0 Z2\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z0 Z3\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z1 Z2\")\n", " + 0.1739537877649417 * of.QubitOperator(\"Z1 Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z2 Z3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 Y2 Y3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 Y2 Y3\")\n", ")\n", "nuclear_repulsion_energy = 0.70556961456"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["pauli_sym = {\"I\": Pauli.I, \"X\": Pauli.X, \"Y\": Pauli.Y, \"Z\": Pauli.Z}"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qps_from_openfermion(paulis):\n", " \"\"\"Convert OpenFermion tensor of Paulis to pytket QubitPauliString.\"\"\"\n", " qlist = []\n", " plist = []\n", " for q, p in paulis:\n", " qlist.append(Qubit(q))\n", " plist.append(pauli_sym[p])\n", " return QubitPauliString(qlist, plist)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qpo_from_openfermion(openf_op):\n", " \"\"\"Convert OpenFermion QubitOperator to pytket QubitPauliOperator.\"\"\"\n", " tk_op = dict()\n", " for term, coeff in openf_op.terms.items():\n", " string = qps_from_openfermion(term)\n", " tk_op[string] = coeff\n", " return QubitPauliOperator(tk_op)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian_op = qpo_from_openfermion(hamiltonian)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Obtain terms for single and double excitations:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["q = [Qubit(i) for i in range(4)]\n", "xyii = QubitPauliString([q[0], q[1]], [Pauli.X, Pauli.Y])\n", "yxii = QubitPauliString([q[0], q[1]], [Pauli.Y, Pauli.X])\n", "iixy = QubitPauliString([q[2], q[3]], [Pauli.X, Pauli.Y])\n", "iiyx = QubitPauliString([q[2], q[3]], [Pauli.Y, Pauli.X])\n", "xxxy = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.X, Pauli.Y])\n", "xxyx = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.Y, Pauli.X])\n", "xyxx = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.X, Pauli.X])\n", "yxxx = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.X, Pauli.X])\n", "yyyx = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.Y, Pauli.X])\n", "yyxy = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.X, Pauli.Y])\n", "yxyy = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.Y, Pauli.Y])\n", "xyyy = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.Y, Pauli.Y])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Symbolic UCC ansatz generation:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["syms = symbols(\"p0 p1 p2\")\n", "singles_syms = {xyii: syms[0], yxii: -syms[0], iixy: syms[1], iiyx: -syms[1]}\n", "doubles_syms = {\n", " xxxy: 0.25 * syms[2],\n", " xxyx: -0.25 * syms[2],\n", " xyxx: 0.25 * syms[2],\n", " yxxx: -0.25 * syms[2],\n", " yyyx: -0.25 * syms[2],\n", " yyxy: 0.25 * syms[2],\n", " yxyy: -0.25 * syms[2],\n", " xyyy: 0.25 * syms[2],\n", "}\n", "excitation_op = QubitPauliOperator({**singles_syms, **doubles_syms})\n", "ucc_ref = Circuit(4).X(0).X(2)\n", "ucc = gen_term_sequence_circuit(excitation_op, ucc_ref)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Circuit simplification:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["GuidedPauliSimp().apply(ucc)\n", "FullPeepholeOptimise().apply(ucc)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Connect to a simulator/device:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc.copy()\n", " sym_map = dict(zip(syms, params))\n", " circ.symbol_substitution(sym_map)\n", " return (\n", " get_operator_expectation_value(\n", " circ,\n", " hamiltonian_op,\n", " backend,\n", " n_shots=4000,\n", " partition_strat=PauliPartitionStrat.CommutingSets,\n", " )\n", " + nuclear_repulsion_energy\n", " ).real"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Optimise against the objective function:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["initial_params = [1e-4, 1e-4, 4e-1]\n", "# #result = minimize(objective, initial_params, method=\"Nelder-Mead\")\n", "# #print(\"Final parameter values\", result.x)\n", "# #print(\"Final energy value\", result.fun)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Exercises:
\n", "- Replace the `get_operator_expectation_value` call with its implementation and use this to pull the analysis for measurement reduction outside of the objective function, so our circuits can be fully determined and compiled once. This means that the `symbol_substitution` method will need to be applied to each measurement circuit instead of just the state preparation circuit.
\n", "- Use the `SpamCorrecter` class to add some mitigation of the measurement errors. Start by running the characterisation circuits first, before your main VQE loop, then apply the mitigation to each of the circuits run within the objective function.
\n", "- Change the `backend` by passing in a `Qiskit` `NoiseModel` to simulate a noisy device. Compare the accuracy of the objective function both with and without the circuit simplification. Try running a classical optimiser over the objective function and compare the convergence rates with different noise models. If you have access to a QPU, try changing the `backend` to connect to that and compare the results to the simulator."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# VQE with UCC ansatz using TKET"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In this tutorial, we will focus on:
\n", "- building parameterised ans\u00e4tze for variational algorithms;
\n", "- compilation tools for UCC-style ans\u00e4tze."]}, {"cell_type": "markdown", "metadata": {}, "source": ["This example assumes the reader is familiar with the Variational Quantum Eigensolver and its application to electronic structure problems through the Unitary Coupled Cluster approach.
\n", "
\n", "To run this example, you will need `pytket` and `pytket-qiskit`, as well as `openfermion`, `scipy`, and `sympy`.
\n", "
\n", "We will start with a basic implementation and then gradually modify it to make it faster, more general, and less noisy. The final solution is given in full at the bottom of the notebook.
\n", "
\n", "Suppose we have some electronic configuration problem, expressed via a physical Hamiltonian. (The Hamiltonian and excitations in this example were obtained using `qiskit-aqua` version 0.5.2 and `pyscf` for H2, bond length 0.75A, sto3g basis, Jordan-Wigner encoding, with no qubit reduction or orbital freezing.). We express it succinctly using the openfermion library:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import openfermion as of"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian = (\n", " -0.8153001706270075 * of.QubitOperator(\"\")\n", " + 0.16988452027940318 * of.QubitOperator(\"Z0\")\n", " + -0.21886306781219608 * of.QubitOperator(\"Z1\")\n", " + 0.16988452027940323 * of.QubitOperator(\"Z2\")\n", " + -0.2188630678121961 * of.QubitOperator(\"Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z0 Z1\")\n", " + 0.16821198673715723 * of.QubitOperator(\"Z0 Z2\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z0 Z3\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z1 Z2\")\n", " + 0.1739537877649417 * of.QubitOperator(\"Z1 Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z2 Z3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 Y2 Y3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 Y2 Y3\")\n", ")\n", "nuclear_repulsion_energy = 0.70556961456"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We would like to define our ansatz for arbitrary parameter values. For simplicity, let's start with a Hardware Efficient Ansatz."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Hardware efficient ansatz:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def hea(params):\n", " ansatz = Circuit(4)\n", " for i in range(4):\n", " ansatz.Ry(params[i], i)\n", " for i in range(3):\n", " ansatz.CX(i, i + 1)\n", " for i in range(4):\n", " ansatz.Ry(params[4 + i], i)\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can use this to build the objective function for our optimisation."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import AerBackend\n", "from pytket.utils.expectations import expectation_from_counts"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Naive objective function:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " energy = 0\n", " for term, coeff in hamiltonian.terms.items():\n", " if not term:\n", " energy += coeff\n", " continue\n", " circ = hea(params)\n", " circ.add_c_register(\"c\", len(term))\n", " for i, (q, pauli) in enumerate(term):\n", " if pauli == \"X\":\n", " circ.H(q)\n", " elif pauli == \"Y\":\n", " circ.V(q)\n", " circ.Measure(q, i)\n", " compiled_circ = backend.get_compiled_circuit(circ)\n", " counts = backend.run_circuit(compiled_circ, n_shots=4000).get_counts()\n", " energy += coeff * expectation_from_counts(counts)\n", " return energy + nuclear_repulsion_energy"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This objective function is then run through a classical optimiser to find the set of parameter values that minimise the energy of the system. For the sake of example, we will just run this with a single parameter value."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["arg_values = [\n", " -7.31158201e-02,\n", " -1.64514836e-04,\n", " 1.12585591e-03,\n", " -2.58367544e-03,\n", " 1.00006068e00,\n", " -1.19551357e-03,\n", " 9.99963988e-01,\n", " 2.53283285e-03,\n", "]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["energy = objective(arg_values)\n", "print(energy)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The HEA is designed to cram as many orthogonal degrees of freedom into a small circuit as possible to be able to explore a large region of the Hilbert space whilst the circuits themselves can be run with minimal noise. These ans\u00e4tze give virtually-optimal circuits by design, but suffer from an excessive number of variational parameters making convergence slow, barren plateaus where the classical optimiser fails to make progress, and spanning a space where most states lack a physical interpretation. These drawbacks can necessitate adding penalties and may mean that the ansatz cannot actually express the true ground state.
\n", "
\n", "The UCC ansatz, on the other hand, is derived from the electronic configuration. It sacrifices efficiency of the circuit for the guarantee of physical states and the variational parameters all having some meaningful effect, which helps the classical optimisation to converge.
\n", "
\n", "This starts by defining the terms of our single and double excitations. These would usually be generated using the orbital configurations, so we will just use a hard-coded example here for the purposes of demonstration."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import Qubit\n", "from pytket.pauli import Pauli, QubitPauliString"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["q = [Qubit(i) for i in range(4)]\n", "xyii = QubitPauliString([q[0], q[1]], [Pauli.X, Pauli.Y])\n", "yxii = QubitPauliString([q[0], q[1]], [Pauli.Y, Pauli.X])\n", "iixy = QubitPauliString([q[2], q[3]], [Pauli.X, Pauli.Y])\n", "iiyx = QubitPauliString([q[2], q[3]], [Pauli.Y, Pauli.X])\n", "xxxy = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.X, Pauli.Y])\n", "xxyx = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.Y, Pauli.X])\n", "xyxx = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.X, Pauli.X])\n", "yxxx = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.X, Pauli.X])\n", "yyyx = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.Y, Pauli.X])\n", "yyxy = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.X, Pauli.Y])\n", "yxyy = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.Y, Pauli.Y])\n", "xyyy = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.Y, Pauli.Y])"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["singles_a = {xyii: 1.0, yxii: -1.0}\n", "singles_b = {iixy: 1.0, iiyx: -1.0}\n", "doubles = {\n", " xxxy: 0.25,\n", " xxyx: -0.25,\n", " xyxx: 0.25,\n", " yxxx: -0.25,\n", " yyyx: -0.25,\n", " yyxy: 0.25,\n", " yxyy: -0.25,\n", " xyyy: 0.25,\n", "}"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Building the ansatz circuit itself is often done naively by defining the map from each term down to basic gates and then applying it to each term."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def add_operator_term(circuit: Circuit, term: QubitPauliString, angle: float):\n", " qubits = []\n", " for q, p in term.map.items():\n", " if p != Pauli.I:\n", " qubits.append(q)\n", " if p == Pauli.X:\n", " circuit.H(q)\n", " elif p == Pauli.Y:\n", " circuit.V(q)\n", " for i in range(len(qubits) - 1):\n", " circuit.CX(i, i + 1)\n", " circuit.Rz(angle, len(qubits) - 1)\n", " for i in reversed(range(len(qubits) - 1)):\n", " circuit.CX(i, i + 1)\n", " for q, p in term.map.items():\n", " if p == Pauli.X:\n", " circuit.H(q)\n", " elif p == Pauli.Y:\n", " circuit.Vdg(q)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Unitary Coupled Cluster Singles & Doubles ansatz:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def ucc(params):\n", " ansatz = Circuit(4)\n", " # Set initial reference state\n", " ansatz.X(1).X(3)\n", " # Evolve by excitations\n", " for term, coeff in singles_a.items():\n", " add_operator_term(ansatz, term, coeff * params[0])\n", " for term, coeff in singles_b.items():\n", " add_operator_term(ansatz, term, coeff * params[1])\n", " for term, coeff in doubles.items():\n", " add_operator_term(ansatz, term, coeff * params[2])\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is already quite verbose, but `pytket` has a neat shorthand construction for these operator terms using the `PauliExpBox` construction. We can then decompose these into basic gates using the `DecomposeBoxes` compiler pass."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import PauliExpBox\n", "from pytket.passes import DecomposeBoxes"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def add_excitation(circ, term_dict, param):\n", " for term, coeff in term_dict.items():\n", " qubits, paulis = zip(*term.map.items())\n", " pbox = PauliExpBox(paulis, coeff * param)\n", " circ.add_pauliexpbox(pbox, qubits)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["UCC ansatz with syntactic shortcuts:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def ucc(params):\n", " ansatz = Circuit(4)\n", " ansatz.X(1).X(3)\n", " add_excitation(ansatz, singles_a, params[0])\n", " add_excitation(ansatz, singles_b, params[1])\n", " add_excitation(ansatz, doubles, params[2])\n", " DecomposeBoxes().apply(ansatz)\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The objective function can also be simplified using a utility method for constructing the measurement circuits and processing for expectation value calculations. For that, we convert the Hamiltonian to a pytket QubitPauliOperator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["pauli_sym = {\"I\": Pauli.I, \"X\": Pauli.X, \"Y\": Pauli.Y, \"Z\": Pauli.Z}"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qps_from_openfermion(paulis):\n", " \"\"\"Convert OpenFermion tensor of Paulis to pytket QubitPauliString.\"\"\"\n", " qlist = []\n", " plist = []\n", " for q, p in paulis:\n", " qlist.append(Qubit(q))\n", " plist.append(pauli_sym[p])\n", " return QubitPauliString(qlist, plist)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qpo_from_openfermion(openf_op):\n", " \"\"\"Convert OpenFermion QubitOperator to pytket QubitPauliOperator.\"\"\"\n", " tk_op = dict()\n", " for term, coeff in openf_op.terms.items():\n", " string = qps_from_openfermion(term)\n", " tk_op[string] = coeff\n", " return QubitPauliOperator(tk_op)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian_op = qpo_from_openfermion(hamiltonian)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Simplified objective function using utilities:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils.expectations import get_operator_expectation_value"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc(params)\n", " return (\n", " get_operator_expectation_value(circ, hamiltonian_op, backend, n_shots=4000)\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["arg_values = [-3.79002933e-05, 2.42964799e-05, 4.63447157e-01]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["energy = objective(arg_values)\n", "print(energy)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is now the simplest form that this operation can take, but it isn't necessarily the most effective. When we decompose the ansatz circuit into basic gates, it is still very expensive. We can employ some of the circuit simplification passes available in `pytket` to reduce its size and improve fidelity in practice.
\n", "
\n", "A good example is to decompose each `PauliExpBox` into basic gates and then apply `FullPeepholeOptimise`, which defines a compilation strategy utilising all of the simplifications in `pytket` that act locally on small regions of a circuit. We can examine the effectiveness by looking at the number of two-qubit gates before and after simplification, which tends to be a good indicator of fidelity for near-term systems where these gates are often slow and inaccurate."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import OpType\n", "from pytket.passes import FullPeepholeOptimise"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_circuit = ucc(arg_values)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count before\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth before\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["FullPeepholeOptimise().apply(test_circuit)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count after FPO\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth after FPO\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These simplification techniques are very general and are almost always beneficial to apply to a circuit if you want to eliminate local redundancies. But UCC ans\u00e4tze have extra structure that we can exploit further. They are defined entirely out of exponentiated tensors of Pauli matrices, giving the regular structure described by the `PauliExpBox`es. Under many circumstances, it is more efficient to not synthesise these constructions individually, but simultaneously in groups. The `PauliSimp` pass finds the description of a given circuit as a sequence of `PauliExpBox`es and resynthesises them (by default, in groups of commuting terms). This can cause great change in the overall structure and shape of the circuit, enabling the identification and elimination of non-local redundancy."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.passes import PauliSimp"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_circuit = ucc(arg_values)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count before\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth before\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["PauliSimp().apply(test_circuit)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count after PS\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth after PS\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["FullPeepholeOptimise().apply(test_circuit)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count after PS+FPO\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth after PS+FPO\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["To include this into our routines, we can just add the simplification passes to the objective function. The `get_operator_expectation_value` utility handles compiling to meet the requirements of the backend, so we don't have to worry about that here."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function with circuit simplification:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc(params)\n", " PauliSimp().apply(circ)\n", " FullPeepholeOptimise().apply(circ)\n", " return (\n", " get_operator_expectation_value(circ, hamiltonian_op, backend, n_shots=4000)\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These circuit simplification techniques have tried to preserve the exact unitary of the circuit, but there are ways to change the unitary whilst preserving the correctness of the algorithm as a whole.
\n", "
\n", "For example, the excitation terms are generated by trotterisation of the excitation operator, and the order of the terms does not change the unitary in the limit of many trotter steps, so in this sense we are free to sequence the terms how we like and it is sensible to do this in a way that enables efficient synthesis of the circuit. Prioritising collecting terms into commuting sets is a very beneficial heuristic for this and can be performed using the `gen_term_sequence_circuit` method to group the terms together into collections of `PauliExpBox`es and the `GuidedPauliSimp` pass to utilise these sets for synthesis."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.passes import GuidedPauliSimp\n", "from pytket.utils import gen_term_sequence_circuit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def ucc(params):\n", " singles_params = {qps: params[0] * coeff for qps, coeff in singles.items()}\n", " doubles_params = {qps: params[1] * coeff for qps, coeff in doubles.items()}\n", " excitation_op = QubitPauliOperator({**singles_params, **doubles_params})\n", " reference_circ = Circuit(4).X(1).X(3)\n", " ansatz = gen_term_sequence_circuit(excitation_op, reference_circ)\n", " GuidedPauliSimp().apply(ansatz)\n", " FullPeepholeOptimise().apply(ansatz)\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Adding these simplification routines doesn't come for free. Compiling and simplifying the circuit to achieve the best results possible can be a difficult task, which can take some time for the classical computer to perform.
\n", "
\n", "During a VQE run, we will call this objective function many times and run many measurement circuits within each, but the circuits that are run on the quantum computer are almost identical, having the same gate structure but with different gate parameters and measurements. We have already exploited this within the body of the objective function by simplifying the ansatz circuit before we call `get_operator_expectation_value`, so it is only done once per objective calculation rather than once per measurement circuit.
\n", "
\n", "We can go even further by simplifying it once outside of the objective function, and then instantiating the simplified ansatz with the parameter values needed. For this, we will construct the UCC ansatz circuit using symbolic (parametric) gates."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from sympy import symbols"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Symbolic UCC ansatz generation:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["syms = symbols(\"p0 p1 p2\")\n", "singles_a_syms = {qps: syms[0] * coeff for qps, coeff in singles_a.items()}\n", "singles_b_syms = {qps: syms[1] * coeff for qps, coeff in singles_b.items()}\n", "doubles_syms = {qps: syms[2] * coeff for qps, coeff in doubles.items()}\n", "excitation_op = QubitPauliOperator({**singles_a_syms, **singles_b_syms, **doubles_syms})\n", "ucc_ref = Circuit(4).X(1).X(3)\n", "ucc = gen_term_sequence_circuit(excitation_op, ucc_ref)\n", "GuidedPauliSimp().apply(ucc)\n", "FullPeepholeOptimise().apply(ucc)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function using the symbolic ansatz:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc.copy()\n", " sym_map = dict(zip(syms, params))\n", " circ.symbol_substitution(sym_map)\n", " return (\n", " get_operator_expectation_value(circ, hamiltonian_op, backend, n_shots=4000)\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We have now got some very good use of `pytket` for simplifying each individual circuit used in our experiment and for minimising the amount of time spent compiling, but there is still more we can do in terms of reducing the amount of work the quantum computer has to do. Currently, each (non-trivial) term in our measurement hamiltonian is measured by a different circuit within each expectation value calculation. Measurement reduction techniques exist for identifying when these observables commute and hence can be simultaneously measured, reducing the number of circuits required for the full expectation value calculation.
\n", "
\n", "This is built in to the `get_operator_expectation_value` method and can be applied by specifying a way to partition the measuremrnt terms. `PauliPartitionStrat.CommutingSets` can greatly reduce the number of measurement circuits by combining any number of terms that mutually commute. However, this involves potentially adding an arbitrary Clifford circuit to change the basis of the measurements which can be costly on NISQ devices, so `PauliPartitionStrat.NonConflictingSets` trades off some of the reduction in circuit number to guarantee that only single-qubit gates are introduced."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.partition import PauliPartitionStrat"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function using measurement reduction:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc.copy()\n", " sym_map = dict(zip(syms, params))\n", " circ.symbol_substitution(sym_map)\n", " return (\n", " get_operator_expectation_value(\n", " circ,\n", " operator,\n", " backend,\n", " n_shots=4000,\n", " partition_strat=PauliPartitionStrat.CommutingSets,\n", " )\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "markdown", "metadata": {}, "source": ["At this point, we have completely transformed how our VQE objective function works, improving its resilience to noise, cutting the number of circuits run, and maintaining fast runtimes. In doing this, we have explored a number of the features `pytket` offers that are beneficial to VQE and the UCC method:
\n", "- high-level syntactic constructs for evolution operators;
\n", "- utility methods for easy expectation value calculations;
\n", "- both generic and domain-specific circuit simplification methods;
\n", "- symbolic circuit compilation;
\n", "- measurement reduction for expectation value calculations."]}, {"cell_type": "markdown", "metadata": {}, "source": ["For the sake of completeness, the following gives the full code for the final solution, including passing the objective function to a classical optimiser to find the ground state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import openfermion as of\n", "from scipy.optimize import minimize\n", "from sympy import symbols"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import AerBackend\n", "from pytket.circuit import Circuit, Qubit\n", "from pytket.partition import PauliPartitionStrat\n", "from pytket.passes import GuidedPauliSimp, FullPeepholeOptimise\n", "from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.utils import get_operator_expectation_value, gen_term_sequence_circuit\n", "from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Obtain electronic Hamiltonian:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian = (\n", " -0.8153001706270075 * of.QubitOperator(\"\")\n", " + 0.16988452027940318 * of.QubitOperator(\"Z0\")\n", " + -0.21886306781219608 * of.QubitOperator(\"Z1\")\n", " + 0.16988452027940323 * of.QubitOperator(\"Z2\")\n", " + -0.2188630678121961 * of.QubitOperator(\"Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z0 Z1\")\n", " + 0.16821198673715723 * of.QubitOperator(\"Z0 Z2\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z0 Z3\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z1 Z2\")\n", " + 0.1739537877649417 * of.QubitOperator(\"Z1 Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z2 Z3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 Y2 Y3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 Y2 Y3\")\n", ")\n", "nuclear_repulsion_energy = 0.70556961456"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["pauli_sym = {\"I\": Pauli.I, \"X\": Pauli.X, \"Y\": Pauli.Y, \"Z\": Pauli.Z}"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qps_from_openfermion(paulis):\n", " \"\"\"Convert OpenFermion tensor of Paulis to pytket QubitPauliString.\"\"\"\n", " qlist = []\n", " plist = []\n", " for q, p in paulis:\n", " qlist.append(Qubit(q))\n", " plist.append(pauli_sym[p])\n", " return QubitPauliString(qlist, plist)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qpo_from_openfermion(openf_op):\n", " \"\"\"Convert OpenFermion QubitOperator to pytket QubitPauliOperator.\"\"\"\n", " tk_op = dict()\n", " for term, coeff in openf_op.terms.items():\n", " string = qps_from_openfermion(term)\n", " tk_op[string] = coeff\n", " return QubitPauliOperator(tk_op)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian_op = qpo_from_openfermion(hamiltonian)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Obtain terms for single and double excitations:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["q = [Qubit(i) for i in range(4)]\n", "xyii = QubitPauliString([q[0], q[1]], [Pauli.X, Pauli.Y])\n", "yxii = QubitPauliString([q[0], q[1]], [Pauli.Y, Pauli.X])\n", "iixy = QubitPauliString([q[2], q[3]], [Pauli.X, Pauli.Y])\n", "iiyx = QubitPauliString([q[2], q[3]], [Pauli.Y, Pauli.X])\n", "xxxy = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.X, Pauli.Y])\n", "xxyx = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.Y, Pauli.X])\n", "xyxx = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.X, Pauli.X])\n", "yxxx = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.X, Pauli.X])\n", "yyyx = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.Y, Pauli.X])\n", "yyxy = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.X, Pauli.Y])\n", "yxyy = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.Y, Pauli.Y])\n", "xyyy = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.Y, Pauli.Y])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Symbolic UCC ansatz generation:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["syms = symbols(\"p0 p1 p2\")\n", "singles_syms = {xyii: syms[0], yxii: -syms[0], iixy: syms[1], iiyx: -syms[1]}\n", "doubles_syms = {\n", " xxxy: 0.25 * syms[2],\n", " xxyx: -0.25 * syms[2],\n", " xyxx: 0.25 * syms[2],\n", " yxxx: -0.25 * syms[2],\n", " yyyx: -0.25 * syms[2],\n", " yyxy: 0.25 * syms[2],\n", " yxyy: -0.25 * syms[2],\n", " xyyy: 0.25 * syms[2],\n", "}\n", "excitation_op = QubitPauliOperator({**singles_syms, **doubles_syms})\n", "ucc_ref = Circuit(4).X(0).X(2)\n", "ucc = gen_term_sequence_circuit(excitation_op, ucc_ref)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Circuit simplification:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["GuidedPauliSimp().apply(ucc)\n", "FullPeepholeOptimise().apply(ucc)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Connect to a simulator/device:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc.copy()\n", " sym_map = dict(zip(syms, params))\n", " circ.symbol_substitution(sym_map)\n", " return (\n", " get_operator_expectation_value(\n", " circ,\n", " hamiltonian_op,\n", " backend,\n", " n_shots=4000,\n", " partition_strat=PauliPartitionStrat.CommutingSets,\n", " )\n", " + nuclear_repulsion_energy\n", " ).real"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Optimise against the objective function:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["initial_params = [1e-4, 1e-4, 4e-1]\n", "# #result = minimize(objective, initial_params, method=\"Nelder-Mead\")\n", "# #print(\"Final parameter values\", result.x)\n", "# #print(\"Final energy value\", result.fun)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Exercises:
\n", "- Replace the `get_operator_expectation_value` call with its implementation and use this to pull the analysis for measurement reduction outside of the objective function, so our circuits can be fully determined and compiled once. This means that the `symbol_substitution` method will need to be applied to each measurement circuit instead of just the state preparation circuit.
\n", "- Use the `SpamCorrecter` class to add some mitigation of the measurement errors. Start by running the characterisation circuits first, before your main VQE loop, then apply the mitigation to each of the circuits run within the objective function.
\n", "- Change the `backend` by passing in a `Qiskit` `NoiseModel` to simulate a noisy device. Compare the accuracy of the objective function both with and without the circuit simplification. Try running a classical optimiser over the objective function and compare the convergence rates with different noise models. If you have access to a QPU, try changing the `backend` to connect to that and compare the results to the simulator."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file From 445ed82d3857288feb75109c576dd708e671a46d Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 26 Oct 2023 18:16:04 +0100 Subject: [PATCH 21/51] modify spam example heading --- examples/python/spam_example.py | 2 +- examples/spam_example.ipynb | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/examples/python/spam_example.py b/examples/python/spam_example.py index 012f2f27..899420fe 100644 --- a/examples/python/spam_example.py +++ b/examples/python/spam_example.py @@ -1,4 +1,4 @@ -# # Calibration and Correction of State Preparation and Measurement (SPAM) +# # Calibration and correction of state preparation and measurement (SPAM) # Quantum Computers available in the NISQ-era are limited by significant sources of device noise which cause errors in computation. One such noise source is errors in the preparation and measurement of quantum states, more commonly know as SPAM. # diff --git a/examples/spam_example.ipynb b/examples/spam_example.ipynb index 543a515b..bcdce3b9 100644 --- a/examples/spam_example.ipynb +++ b/examples/spam_example.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Calibration and Correction of State Preparation and Measurement (SPAM)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Quantum Computers available in the NISQ-era are limited by significant sources of device noise which cause errors in computation. One such noise source is errors in the preparation and measurement of quantum states, more commonly know as SPAM.
\n", "
\n", "If device SPAM error can be characterised, then device results can be modified to mitigate the error. Characterisation proceeds by determining overlap between different prepared basis states when measured, and mitigation modifies the distribution over output states of the corrected circuit. No modification of the quantum circuit being corrected is required. The ``` pytket``` ```SpamCorrecter``` class supports characterisation and mitigation of device SPAM error.
\n", "
\n", "In this tutorial we will show how the ```SpamCorrecter``` class can be used to modify real results and improve device performance when running experiments.
\n", "
\n", "This tutorial will require installation of ```pytket```, ```pytket_qiskit``` and ```qiskit```, all available on pip.
\n", "
\n", "First, import the ```SpamCorrecter``` class."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils.spam import SpamCorrecter"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The SpamCorrecter class has methods for generating State Preparation and Measurement (SPAM) calibration experiments for pytket backends and correcting counts generated from those same backends.
\n", "
\n", "Let's first mitigate error from a noisy simulation, using a noise model straight from the 5-qubit IBMQ manila device. This will require a preloaded IBMQ account."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from qiskit import IBMQ"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["IBMQ.load_account()"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import process_characterisation"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ibmq_manila_backend = IBMQ.providers()[0].get_backend(\"ibmq_manila\")\n", "pytket_manila_characterisation = process_characterisation(ibmq_manila_backend)\n", "pytket_manila_architecture = pytket_manila_characterisation[\"Architecture\"]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import networkx as nx\n", "import matplotlib.pyplot as plt"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["manila_graph = nx.Graph(pytket_manila_architecture.coupling)\n", "nx.draw(manila_graph, labels={node: node for node in manila_graph.nodes()})"]}, {"cell_type": "markdown", "metadata": {}, "source": ["SPAM correction requires subsets of qubits which are assumed to only have SPAM errors correlated with each other, and no other qubits.
\n", "
\n", "Correlated errors are usually dependent on the connectivity layout of devices, as shown above.
\n", "
\n", "As manila is a small 5-qubit device with few connections, let's assume that all qubits have correlated SPAM errors. The number of calibration circuits produced is exponential in the maximum number of correlated circuits, so finding good subsets of correlated qubits is important for characterising larger devices with smaller experimental overhead.
\n", "
\n", "We can produce an ```IBMQEmulatorBackend``` to run this. This uses a noise model from ```ibmq_manila``` produced using qiskit-aer. We can then execute all calibration circuits through the backend."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import IBMQEmulatorBackend, AerBackend"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["n_shots = 8192\n", "pytket_noisy_sim_backend = IBMQEmulatorBackend(\"ibmq_manila\")\n", "manila_node_subsets = pytket_noisy_sim_backend.backend_info.architecture.nodes\n", "manila_spam = SpamCorrecter([manila_node_subsets], pytket_noisy_sim_backend)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The SpamCorrecter uses these subsets of qubits to produce calibration circuits."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["calibration_circuits = manila_spam.calibration_circuits()\n", "print(\"Number of calibration circuits: \", len(calibration_circuits))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["sim_handles = pytket_noisy_sim_backend.process_circuits(calibration_circuits, n_shots)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Count results from the simulator are then used to calculate the matrices used for SPAM correction for ```ibmq_manila```."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["sim_count_results = pytket_noisy_sim_backend.get_results(sim_handles)\n", "manila_spam.calculate_matrices(sim_count_results)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ghz_circuit = (\n", " Circuit(len(pytket_noisy_sim_backend.backend_info.architecture.nodes))\n", " .H(0)\n", " .CX(0, 1)\n", " .CX(1, 2)\n", " .measure_all()\n", ")\n", "ghz_circuit = pytket_noisy_sim_backend.get_compiled_circuit(ghz_circuit)\n", "ghz_noisy_handle = pytket_noisy_sim_backend.process_circuit(ghz_circuit, n_shots)\n", "ghz_noisy_result = pytket_noisy_sim_backend.get_result(ghz_noisy_handle)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We also run a noiseless simulation so we can compare performance."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["pytket_noiseless_sim_backend = AerBackend()\n", "ghz_noiseless_handle = pytket_noiseless_sim_backend.process_circuit(\n", " ghz_circuit, n_shots\n", ")\n", "ghz_noiseless_result = pytket_noiseless_sim_backend.get_result(ghz_noiseless_handle)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Noisy simulator counts are corrected using the ```SpamCorrecter``` objects ```correct_counts``` method.
\n", "
\n", "To correctly amend counts, the ```correct_counts``` method requires a ``ParallelMeasures`` type object, a list of ``Dict[Qubit, Bit]`` where each dictionary denotes a set of Qubit measured in parallel and the Bit their measured values are assigned to.
\n", "
\n", "The ``SpamCorrecter`` class has a helper method ``get_parallel_measure`` for retrieving this object for a Circuit."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ghz_parallel_measure = manila_spam.get_parallel_measure(ghz_circuit)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ghz_spam_corrected_result = manila_spam.correct_counts(\n", " ghz_noisy_result, ghz_parallel_measure\n", ")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Import and define the Jensen-Shannon divergence, which we will use for comparing performance. The Jensen-Shannon divergence is a symmetric and finite measure of similarity between two probability distributions. A smaller divergence implies more similarity between two probability distributions."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from scipy.stats import entropy\n", "import numpy as np\n", "import itertools"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def binseq(k):\n", " return [\"\".join(x) for x in itertools.product(\"01\", repeat=k)]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def probs_from_counts(result):\n", " counts = result.get_counts()\n", " counts_dict = dict()\n", " for x in counts:\n", " counts_dict[\"\".join(str(e) for e in x)] = counts[x]\n", " converted = []\n", " binary_strings = binseq(len(list(counts.keys())[0]))\n", " for b in binary_strings:\n", " converted.append(counts_dict.get(b, 0))\n", " return converted / np.sum(converted)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def JSD(P, Q):\n", " _P = P / np.linalg.norm(P, ord=1)\n", " _Q = Q / np.linalg.norm(Q, ord=1)\n", " _M = 0.5 * (_P + _Q)\n", " return 0.5 * (entropy(_P, _M) + entropy(_Q, _M))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Convert our counts results to a probability distribution over the basis states for comparison."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ghz_noiseless_probabilities = probs_from_counts(ghz_noiseless_result)\n", "ghz_noisy_probabilities = probs_from_counts(ghz_noisy_result)\n", "ghz_spam_corrected_probabilities = probs_from_counts(ghz_spam_corrected_result)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\n", " \"Jensen-Shannon Divergence between noiseless simulation probability distribution and noisy simulation probability distribution: \",\n", " JSD(ghz_noiseless_probabilities, ghz_noisy_probabilities),\n", ")\n", "print(\n", " \"Jensen-Shannon Divergence between noiseless simulation probability distribution and spam corrected noisy simulation probability distribution: \",\n", " JSD(ghz_noiseless_probabilities, ghz_spam_corrected_probabilities),\n", ")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In our noisy simulated case, spam corrected results produced a distribution closer to the expected distribution.
\n", "
\n", "There are two methods available for correcting counts: the default ```bayesian```, and ```invert```. Further information on each method is available at our [documentation](https://cqcl.github.io/tket/pytket/api/utils.html#module-pytket.utils.spam).
\n", "
\n", "Let's look at how the ```invert``` method performs."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ghz_invert_corrected_result = manila_spam.correct_counts(\n", " ghz_noisy_result, ghz_parallel_measure, method=\"invert\"\n", ")\n", "ghz_invert_probabilities = probs_from_counts(ghz_invert_corrected_result)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\n", " \"Jensen-Shannon Divergence between noiseless simulation probability distribution and Bayesian-corrected noisy simulation probability distribution: \",\n", " JSD(ghz_noiseless_probabilities, ghz_spam_corrected_probabilities),\n", ")\n", "print(\n", " \"Jensen-Shannon Divergence between noiseless simulation probability distribution and invert-corrected noisy simulation probability distribution: \",\n", " JSD(ghz_noiseless_probabilities, ghz_invert_probabilities),\n", ")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["To see how SPAM correction performs on results from a real IBMQ quantum device, try replacing `IBMQEmulatorBackend` with `IBMQBackend`."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import IBMQBackend"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ibm_backend = IBMQBackend(\"ibmq_manila\")"]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Calibration and correction of state preparation and measurement (SPAM)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Quantum Computers available in the NISQ-era are limited by significant sources of device noise which cause errors in computation. One such noise source is errors in the preparation and measurement of quantum states, more commonly know as SPAM.
\n", "
\n", "If device SPAM error can be characterised, then device results can be modified to mitigate the error. Characterisation proceeds by determining overlap between different prepared basis states when measured, and mitigation modifies the distribution over output states of the corrected circuit. No modification of the quantum circuit being corrected is required. The ``` pytket``` ```SpamCorrecter``` class supports characterisation and mitigation of device SPAM error.
\n", "
\n", "In this tutorial we will show how the ```SpamCorrecter``` class can be used to modify real results and improve device performance when running experiments.
\n", "
\n", "This tutorial will require installation of ```pytket```, ```pytket_qiskit``` and ```qiskit```, all available on pip.
\n", "
\n", "First, import the ```SpamCorrecter``` class."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils.spam import SpamCorrecter"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The SpamCorrecter class has methods for generating State Preparation and Measurement (SPAM) calibration experiments for pytket backends and correcting counts generated from those same backends.
\n", "
\n", "Let's first mitigate error from a noisy simulation, using a noise model straight from the 5-qubit IBMQ manila device. This will require a preloaded IBMQ account."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from qiskit import IBMQ"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["IBMQ.load_account()"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import process_characterisation"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ibmq_manila_backend = IBMQ.providers()[0].get_backend(\"ibmq_manila\")\n", "pytket_manila_characterisation = process_characterisation(ibmq_manila_backend)\n", "pytket_manila_architecture = pytket_manila_characterisation[\"Architecture\"]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import networkx as nx\n", "import matplotlib.pyplot as plt"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["manila_graph = nx.Graph(pytket_manila_architecture.coupling)\n", "nx.draw(manila_graph, labels={node: node for node in manila_graph.nodes()})"]}, {"cell_type": "markdown", "metadata": {}, "source": ["SPAM correction requires subsets of qubits which are assumed to only have SPAM errors correlated with each other, and no other qubits.
\n", "
\n", "Correlated errors are usually dependent on the connectivity layout of devices, as shown above.
\n", "
\n", "As manila is a small 5-qubit device with few connections, let's assume that all qubits have correlated SPAM errors. The number of calibration circuits produced is exponential in the maximum number of correlated circuits, so finding good subsets of correlated qubits is important for characterising larger devices with smaller experimental overhead.
\n", "
\n", "We can produce an ```IBMQEmulatorBackend``` to run this. This uses a noise model from ```ibmq_manila``` produced using qiskit-aer. We can then execute all calibration circuits through the backend."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import IBMQEmulatorBackend, AerBackend"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["n_shots = 8192\n", "pytket_noisy_sim_backend = IBMQEmulatorBackend(\"ibmq_manila\")\n", "manila_node_subsets = pytket_noisy_sim_backend.backend_info.architecture.nodes\n", "manila_spam = SpamCorrecter([manila_node_subsets], pytket_noisy_sim_backend)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The SpamCorrecter uses these subsets of qubits to produce calibration circuits."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["calibration_circuits = manila_spam.calibration_circuits()\n", "print(\"Number of calibration circuits: \", len(calibration_circuits))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["sim_handles = pytket_noisy_sim_backend.process_circuits(calibration_circuits, n_shots)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Count results from the simulator are then used to calculate the matrices used for SPAM correction for ```ibmq_manila```."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["sim_count_results = pytket_noisy_sim_backend.get_results(sim_handles)\n", "manila_spam.calculate_matrices(sim_count_results)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ghz_circuit = (\n", " Circuit(len(pytket_noisy_sim_backend.backend_info.architecture.nodes))\n", " .H(0)\n", " .CX(0, 1)\n", " .CX(1, 2)\n", " .measure_all()\n", ")\n", "ghz_circuit = pytket_noisy_sim_backend.get_compiled_circuit(ghz_circuit)\n", "ghz_noisy_handle = pytket_noisy_sim_backend.process_circuit(ghz_circuit, n_shots)\n", "ghz_noisy_result = pytket_noisy_sim_backend.get_result(ghz_noisy_handle)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We also run a noiseless simulation so we can compare performance."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["pytket_noiseless_sim_backend = AerBackend()\n", "ghz_noiseless_handle = pytket_noiseless_sim_backend.process_circuit(\n", " ghz_circuit, n_shots\n", ")\n", "ghz_noiseless_result = pytket_noiseless_sim_backend.get_result(ghz_noiseless_handle)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Noisy simulator counts are corrected using the ```SpamCorrecter``` objects ```correct_counts``` method.
\n", "
\n", "To correctly amend counts, the ```correct_counts``` method requires a ``ParallelMeasures`` type object, a list of ``Dict[Qubit, Bit]`` where each dictionary denotes a set of Qubit measured in parallel and the Bit their measured values are assigned to.
\n", "
\n", "The ``SpamCorrecter`` class has a helper method ``get_parallel_measure`` for retrieving this object for a Circuit."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ghz_parallel_measure = manila_spam.get_parallel_measure(ghz_circuit)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ghz_spam_corrected_result = manila_spam.correct_counts(\n", " ghz_noisy_result, ghz_parallel_measure\n", ")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Import and define the Jensen-Shannon divergence, which we will use for comparing performance. The Jensen-Shannon divergence is a symmetric and finite measure of similarity between two probability distributions. A smaller divergence implies more similarity between two probability distributions."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from scipy.stats import entropy\n", "import numpy as np\n", "import itertools"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def binseq(k):\n", " return [\"\".join(x) for x in itertools.product(\"01\", repeat=k)]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def probs_from_counts(result):\n", " counts = result.get_counts()\n", " counts_dict = dict()\n", " for x in counts:\n", " counts_dict[\"\".join(str(e) for e in x)] = counts[x]\n", " converted = []\n", " binary_strings = binseq(len(list(counts.keys())[0]))\n", " for b in binary_strings:\n", " converted.append(counts_dict.get(b, 0))\n", " return converted / np.sum(converted)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def JSD(P, Q):\n", " _P = P / np.linalg.norm(P, ord=1)\n", " _Q = Q / np.linalg.norm(Q, ord=1)\n", " _M = 0.5 * (_P + _Q)\n", " return 0.5 * (entropy(_P, _M) + entropy(_Q, _M))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Convert our counts results to a probability distribution over the basis states for comparison."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ghz_noiseless_probabilities = probs_from_counts(ghz_noiseless_result)\n", "ghz_noisy_probabilities = probs_from_counts(ghz_noisy_result)\n", "ghz_spam_corrected_probabilities = probs_from_counts(ghz_spam_corrected_result)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\n", " \"Jensen-Shannon Divergence between noiseless simulation probability distribution and noisy simulation probability distribution: \",\n", " JSD(ghz_noiseless_probabilities, ghz_noisy_probabilities),\n", ")\n", "print(\n", " \"Jensen-Shannon Divergence between noiseless simulation probability distribution and spam corrected noisy simulation probability distribution: \",\n", " JSD(ghz_noiseless_probabilities, ghz_spam_corrected_probabilities),\n", ")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In our noisy simulated case, spam corrected results produced a distribution closer to the expected distribution.
\n", "
\n", "There are two methods available for correcting counts: the default ```bayesian```, and ```invert```. Further information on each method is available at our [documentation](https://cqcl.github.io/tket/pytket/api/utils.html#module-pytket.utils.spam).
\n", "
\n", "Let's look at how the ```invert``` method performs."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ghz_invert_corrected_result = manila_spam.correct_counts(\n", " ghz_noisy_result, ghz_parallel_measure, method=\"invert\"\n", ")\n", "ghz_invert_probabilities = probs_from_counts(ghz_invert_corrected_result)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\n", " \"Jensen-Shannon Divergence between noiseless simulation probability distribution and Bayesian-corrected noisy simulation probability distribution: \",\n", " JSD(ghz_noiseless_probabilities, ghz_spam_corrected_probabilities),\n", ")\n", "print(\n", " \"Jensen-Shannon Divergence between noiseless simulation probability distribution and invert-corrected noisy simulation probability distribution: \",\n", " JSD(ghz_noiseless_probabilities, ghz_invert_probabilities),\n", ")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["To see how SPAM correction performs on results from a real IBMQ quantum device, try replacing `IBMQEmulatorBackend` with `IBMQBackend`."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import IBMQBackend"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["ibm_backend = IBMQBackend(\"ibmq_manila\")"]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file From 9aa2d34fdcd6f14263b3e9714b3aec2f7a91b1a1 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Fri, 27 Oct 2023 14:42:13 +0100 Subject: [PATCH 22/51] add Getting started notebook --- examples/Getting_started.ipynb | 249 +++++++++++++++++++++++++++++++++ examples/_toc.yml | 3 +- 2 files changed, 251 insertions(+), 1 deletion(-) create mode 100644 examples/Getting_started.ipynb diff --git a/examples/Getting_started.ipynb b/examples/Getting_started.ipynb new file mode 100644 index 00000000..bc3b37dd --- /dev/null +++ b/examples/Getting_started.ipynb @@ -0,0 +1,249 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "id": "f5f6b50c-9dbc-4a85-afe2-1b1ff717d091", + "metadata": {}, + "source": [ + "# pytket example notebooks - Getting started" + ] + }, + { + "cell_type": "markdown", + "id": "e6d90a8a-4303-4d6d-9e51-ca8fcec0a2a5", + "metadata": {}, + "source": [ + "## How do I build a circuit?\n", + "\n", + "### Using the `Circuit` class directly\n", + "\n", + "You can create a circuit by creating an instance of the `Circuit` class and adding gates manually." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "32a81360-0002-4c82-9a9e-a0f90851f34b", + "metadata": {}, + "outputs": [], + "source": [ + "from pytket import Circuit\n", + "\n", + "ghz_circ = Circuit(3)\n", + "ghz_circ.H(0)\n", + "ghz_circ.CX(0, 1)\n", + "ghz_circ.CX(1, 2)\n", + "ghz_circ.add_barrier(ghz_circ.qubits)\n", + "ghz_circ.measure_all()" + ] + }, + { + "cell_type": "markdown", + "id": "147c590e-89e5-4e4e-9fb9-d10ac5689fec", + "metadata": {}, + "source": [ + "Now let's draw a nice picture of the circuit with the circuit renderer" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "92df4f4d-487c-4c56-a072-9a1d2e3c07fc", + "metadata": {}, + "outputs": [], + "source": [ + "from pytket.circuit.display import render_circuit_jupyter\n", + "\n", + "render_circuit_jupyter(ghz_circ)" + ] + }, + { + "cell_type": "markdown", + "id": "bdec557e", + "metadata": {}, + "source": [ + "See also the [Circuit construction](https://tket.quantinuum.com/user-manual/manual_circuit.html) section of the user manual." + ] + }, + { + "cell_type": "markdown", + "id": "820b0215-4ca8-450c-bc1a-62bed65e2740", + "metadata": {}, + "source": [ + "### Using a QASM file" + ] + }, + { + "cell_type": "markdown", + "id": "7c9089dc-cf4d-4efd-b90e-5485218f90a5", + "metadata": {}, + "source": [ + "Alternatively we can import a circuit from a QASM file using [pytket.qasm](https://tket.quantinuum.com/api-docs/qasm.html). There are also functions for generating a circuit from a QASM string or exporting to a qasm file.\n", + "\n", + "\n", + "Note that its also possible to import a circuit from quipper using [pytket.quipper](https://tket.quantinuum.com/api-docs/quipper.html) module." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "1bf32877-f5e0-4b09-a12d-7aa8fa7d5d20", + "metadata": {}, + "outputs": [], + "source": [ + "from pytket.qasm import circuit_from_qasm\n", + "\n", + "w_state_circ = circuit_from_qasm(\"qasm/W-state.qasm\")\n", + "render_circuit_jupyter(w_state_circ)" + ] + }, + { + "cell_type": "markdown", + "id": "ef0c31c5-9203-475b-abe1-5a0b53dc5e87", + "metadata": {}, + "source": [ + "## From qiskit (or other quantum computing library)\n", + "\n", + "Its possible to generate a circuit directly from a qiskit `QuantumCircuit` using the [qiskit_to_tk](feature/control_state_improvement) function." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "1906b037-2b1c-454a-a51a-67350e3f200b", + "metadata": {}, + "outputs": [], + "source": [ + "from qiskit import QuantumCircuit\n", + "\n", + "qiskit_circ = QuantumCircuit(4)\n", + "qiskit_circ.h(range(4))\n", + "qiskit_circ.ccx(0, 2, 1)\n", + "qiskit_circ.cry(0.25, 1, 2)\n", + "qiskit_circ.crz(0.15, 2, 0)\n", + "qiskit_circ.ccx(2, 1, 3)\n", + "print(qiskit_circ)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "6cec62fe-f872-45d8-8c2b-6c97f0613854", + "metadata": {}, + "outputs": [], + "source": [ + "from pytket.extensions.qiskit import qiskit_to_tk\n", + "\n", + "tket_circ = qiskit_to_tk(qiskit_circ)\n", + "render_circuit_jupyter(tket_circ)" + ] + }, + { + "cell_type": "markdown", + "id": "7688630f-89e2-48c7-9d1e-53a9dedf088a", + "metadata": {}, + "source": [ + "Note that although these circuits look the same they are not unitarily equiavelent as pytket and qiskit use different qubit ordering conventions.\n", + "\n", + "pytket uses little endian whereas qiskit uses the big-endian ordering convention." + ] + }, + { + "cell_type": "markdown", + "id": "1eee94cb-d7a2-469d-9c48-747a6e9271a1", + "metadata": {}, + "source": [ + "Circuit conversion functions are also available for [pytket-cirq](https://tket.quantinuum.com/extensions/pytket-cirq/), [pytket-pennylane](https://tket.quantinuum.com/extensions/pytket-pennylane/), [pytket-braket](https://tket.quantinuum.com/extensions/pytket-braket/) and more." + ] + }, + { + "cell_type": "markdown", + "id": "ae85113b-5f8c-43b0-93c8-dc89002deece", + "metadata": {}, + "source": [ + "## Using Backends" + ] + }, + { + "cell_type": "markdown", + "id": "f2684ec5-30d9-4003-ad6a-624ec20bd268", + "metadata": {}, + "source": [ + "In pytket a `Backend` represents an interface to a quantum device or simulator.\n", + "\n", + "We will show a simple example of running the `ghz_circ` defined above on the `AerBackend` simulator." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "42f4e1f6-2281-4780-bb11-7132c4be5313", + "metadata": {}, + "outputs": [], + "source": [ + "render_circuit_jupyter(ghz_circ)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "868189e3-197c-427b-843e-6a991d1cfec9", + "metadata": {}, + "outputs": [], + "source": [ + "from pytket.extensions.qiskit import AerBackend\n", + "\n", + "backend = AerBackend()\n", + "result = backend.run_circuit(ghz_circ)\n", + "print(result.get_counts())" + ] + }, + { + "cell_type": "markdown", + "id": "eadff4bc-a1d3-42d0-bc0e-6bf515f40715", + "metadata": {}, + "source": [ + "The `AerBackend` simulator is highly idealised having a broad gateset, and no restrictive connectivity or device noise.\n", + "\n", + "The Hadamard and CX gate are supported operations of the simulator so we can run the GHZ circuit without changing any of the operations. For more realistic cases a compiler will have to solve for the limited gateset of the target backend as well as other backend requirements." + ] + }, + { + "cell_type": "markdown", + "id": "3321855a-683b-4848-81bd-1d238626f45b", + "metadata": {}, + "source": [ + "See the [Running on Backends](https://tket.quantinuum.com/user-manual/manual_backend.html) section of the user manual and the [backends example notebook](https://tket.quantinuum.com/examples/backends_example.html) for more." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "0e55e2b6-3ba8-47d1-a0b9-b10d2308c4e7", + "metadata": {}, + "outputs": [], + "source": [] + } + ], + "metadata": { + "kernelspec": { + "display_name": "Python 3 (ipykernel)", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.10.6" + } + }, + "nbformat": 4, + "nbformat_minor": 5 +} diff --git a/examples/_toc.yml b/examples/_toc.yml index 692002ff..a9532240 100644 --- a/examples/_toc.yml +++ b/examples/_toc.yml @@ -2,7 +2,7 @@ # Learn more at https://jupyterbook.org/customize/toc.html format: jb-book -root: README +root: Getting_started parts: - caption: Building Quantum Circuits chapters: @@ -37,3 +37,4 @@ parts: - file: entanglement_swapping - file: spam_example - file: expectation_value_example + - file: README From e68cc9a54017bb296fe0d59a46f6ae86403b40d0 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Fri, 27 Oct 2023 14:44:23 +0100 Subject: [PATCH 23/51] fix reference --- examples/Getting_started.ipynb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/examples/Getting_started.ipynb b/examples/Getting_started.ipynb index bc3b37dd..97c65b5a 100644 --- a/examples/Getting_started.ipynb +++ b/examples/Getting_started.ipynb @@ -104,7 +104,7 @@ "source": [ "## From qiskit (or other quantum computing library)\n", "\n", - "Its possible to generate a circuit directly from a qiskit `QuantumCircuit` using the [qiskit_to_tk](feature/control_state_improvement) function." + "Its possible to generate a circuit directly from a qiskit `QuantumCircuit` using the [qiskit_to_tk](https://tket.quantinuum.com/extensions/pytket-qiskit/api.html#pytket.extensions.qiskit.tk_to_qiskit) function." ] }, { From 90161129ea780f40a4ff336eb930df248503d9e0 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Fri, 27 Oct 2023 14:45:59 +0100 Subject: [PATCH 24/51] change heading --- examples/Getting_started.ipynb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/examples/Getting_started.ipynb b/examples/Getting_started.ipynb index 97c65b5a..d87f88db 100644 --- a/examples/Getting_started.ipynb +++ b/examples/Getting_started.ipynb @@ -5,7 +5,7 @@ "id": "f5f6b50c-9dbc-4a85-afe2-1b1ff717d091", "metadata": {}, "source": [ - "# pytket example notebooks - Getting started" + "# pytket examples" ] }, { From 6ce001e5dbf30004effe5e1d336d1de9d072f6fc Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Fri, 27 Oct 2023 14:48:16 +0100 Subject: [PATCH 25/51] fix README --- examples/README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/examples/README.md b/examples/README.md index b9948ce0..fb7b09cc 100644 --- a/examples/README.md +++ b/examples/README.md @@ -1,4 +1,4 @@ -# pytket examples +# Contributing new notebooks See the pytket examples built with jupyterbook [here](https://tket.quantinuum.com/examples). From 7c6fb4a65ea73695c50e141b43578068d0a84ccd Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Fri, 27 Oct 2023 14:50:28 +0100 Subject: [PATCH 26/51] more fixes --- examples/Getting_started.ipynb | 2 +- examples/_config.yml | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/examples/Getting_started.ipynb b/examples/Getting_started.ipynb index d87f88db..498ee7d2 100644 --- a/examples/Getting_started.ipynb +++ b/examples/Getting_started.ipynb @@ -102,7 +102,7 @@ "id": "ef0c31c5-9203-475b-abe1-5a0b53dc5e87", "metadata": {}, "source": [ - "## From qiskit (or other quantum computing library)\n", + "### From qiskit (or other quantum computing library)\n", "\n", "Its possible to generate a circuit directly from a qiskit `QuantumCircuit` using the [qiskit_to_tk](https://tket.quantinuum.com/extensions/pytket-qiskit/api.html#pytket.extensions.qiskit.tk_to_qiskit) function." ] diff --git a/examples/_config.yml b/examples/_config.yml index b5431aa5..0e2e6bf7 100644 --- a/examples/_config.yml +++ b/examples/_config.yml @@ -19,7 +19,7 @@ execute: repository: url: https://github.com/CQCL/pytket # Notebook files are located in pytket/examples path_to_book: docs # Optional path to your book, relative to the repository root - branch: master # Which branch of the repository should be used when creating links (optional) + branch: main # Which branch of the repository should be used when creating links (optional) # Add GitHub buttons to your book # See https://jupyterbook.org/customize/config.html#add-a-link-to-your-repository From ef5fd654b9a1766fa2517fd94f4afc3b9d88efe2 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Mon, 30 Oct 2023 10:32:18 +0000 Subject: [PATCH 27/51] Update getting started notebook --- examples/Getting_started.ipynb | 39 ---------------------------------- 1 file changed, 39 deletions(-) diff --git a/examples/Getting_started.ipynb b/examples/Getting_started.ipynb index 498ee7d2..f8bc8838 100644 --- a/examples/Getting_started.ipynb +++ b/examples/Getting_started.ipynb @@ -107,37 +107,6 @@ "Its possible to generate a circuit directly from a qiskit `QuantumCircuit` using the [qiskit_to_tk](https://tket.quantinuum.com/extensions/pytket-qiskit/api.html#pytket.extensions.qiskit.tk_to_qiskit) function." ] }, - { - "cell_type": "code", - "execution_count": null, - "id": "1906b037-2b1c-454a-a51a-67350e3f200b", - "metadata": {}, - "outputs": [], - "source": [ - "from qiskit import QuantumCircuit\n", - "\n", - "qiskit_circ = QuantumCircuit(4)\n", - "qiskit_circ.h(range(4))\n", - "qiskit_circ.ccx(0, 2, 1)\n", - "qiskit_circ.cry(0.25, 1, 2)\n", - "qiskit_circ.crz(0.15, 2, 0)\n", - "qiskit_circ.ccx(2, 1, 3)\n", - "print(qiskit_circ)" - ] - }, - { - "cell_type": "code", - "execution_count": null, - "id": "6cec62fe-f872-45d8-8c2b-6c97f0613854", - "metadata": {}, - "outputs": [], - "source": [ - "from pytket.extensions.qiskit import qiskit_to_tk\n", - "\n", - "tket_circ = qiskit_to_tk(qiskit_circ)\n", - "render_circuit_jupyter(tket_circ)" - ] - }, { "cell_type": "markdown", "id": "7688630f-89e2-48c7-9d1e-53a9dedf088a", @@ -215,14 +184,6 @@ "source": [ "See the [Running on Backends](https://tket.quantinuum.com/user-manual/manual_backend.html) section of the user manual and the [backends example notebook](https://tket.quantinuum.com/examples/backends_example.html) for more." ] - }, - { - "cell_type": "code", - "execution_count": null, - "id": "0e55e2b6-3ba8-47d1-a0b9-b10d2308c4e7", - "metadata": {}, - "outputs": [], - "source": [] } ], "metadata": { From b1069a703e88be00a41b44540014a525438296a4 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Mon, 30 Oct 2023 10:49:53 +0000 Subject: [PATCH 28/51] another text edit --- examples/Getting_started.ipynb | 11 ++++------- 1 file changed, 4 insertions(+), 7 deletions(-) diff --git a/examples/Getting_started.ipynb b/examples/Getting_started.ipynb index f8bc8838..1e0c6129 100644 --- a/examples/Getting_started.ipynb +++ b/examples/Getting_started.ipynb @@ -13,9 +13,8 @@ "id": "e6d90a8a-4303-4d6d-9e51-ca8fcec0a2a5", "metadata": {}, "source": [ - "## How do I build a circuit?\n", + "## Building a circuit with the `Circuit` class\n", "\n", - "### Using the `Circuit` class directly\n", "\n", "You can create a circuit by creating an instance of the `Circuit` class and adding gates manually." ] @@ -70,7 +69,7 @@ "id": "820b0215-4ca8-450c-bc1a-62bed65e2740", "metadata": {}, "source": [ - "### Using a QASM file" + "## Build a `Circuit` from a QASM file" ] }, { @@ -102,7 +101,7 @@ "id": "ef0c31c5-9203-475b-abe1-5a0b53dc5e87", "metadata": {}, "source": [ - "### From qiskit (or other quantum computing library)\n", + "## Import a circuit from qiskit (or other quantum computing library)\n", "\n", "Its possible to generate a circuit directly from a qiskit `QuantumCircuit` using the [qiskit_to_tk](https://tket.quantinuum.com/extensions/pytket-qiskit/api.html#pytket.extensions.qiskit.tk_to_qiskit) function." ] @@ -112,9 +111,7 @@ "id": "7688630f-89e2-48c7-9d1e-53a9dedf088a", "metadata": {}, "source": [ - "Note that although these circuits look the same they are not unitarily equiavelent as pytket and qiskit use different qubit ordering conventions.\n", - "\n", - "pytket uses little endian whereas qiskit uses the big-endian ordering convention." + "Note that pytket and qiskit use opposite qubit ordering conventions. So circuits which look identical may correspond to different unitary operations." ] }, { From df1e650eab8fe6fb14e002a8b2c69b3730a58076 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Mon, 30 Oct 2023 10:57:30 +0000 Subject: [PATCH 29/51] add tket -> qiskit example --- examples/Getting_started.ipynb | 401 ++++++++++++++++++++++++++++++++- 1 file changed, 390 insertions(+), 11 deletions(-) diff --git a/examples/Getting_started.ipynb b/examples/Getting_started.ipynb index 1e0c6129..3dc02389 100644 --- a/examples/Getting_started.ipynb +++ b/examples/Getting_started.ipynb @@ -21,10 +21,21 @@ }, { "cell_type": "code", - "execution_count": null, + "execution_count": 1, "id": "32a81360-0002-4c82-9a9e-a0f90851f34b", "metadata": {}, - "outputs": [], + "outputs": [ + { + "data": { + "text/plain": [ + "[H q[0]; CX q[0], q[1]; CX q[1], q[2]; Barrier q[0], q[1], q[2]; Measure q[0] --> c[0]; Measure q[1] --> c[1]; Measure q[2] --> c[2]; ]" + ] + }, + "execution_count": 1, + "metadata": {}, + "output_type": "execute_result" + } + ], "source": [ "from pytket import Circuit\n", "\n", @@ -46,10 +57,89 @@ }, { "cell_type": "code", - "execution_count": null, + "execution_count": 2, "id": "92df4f4d-487c-4c56-a072-9a1d2e3c07fc", "metadata": {}, - "outputs": [], + "outputs": [ + { + "data": { + "text/html": [ + "\n", + "\n", + "\n", + "\n", + "\n", + "\n", + "
\n", + " \n", + "
\n", + "\n" + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + } + ], "source": [ "from pytket.circuit.display import render_circuit_jupyter\n", "\n", @@ -85,10 +175,89 @@ }, { "cell_type": "code", - "execution_count": null, + "execution_count": 3, "id": "1bf32877-f5e0-4b09-a12d-7aa8fa7d5d20", "metadata": {}, - "outputs": [], + "outputs": [ + { + "data": { + "text/html": [ + "\n", + "\n", + "\n", + "\n", + "\n", + "\n", + "
\n", + " \n", + "
\n", + "\n" + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + } + ], "source": [ "from pytket.qasm import circuit_from_qasm\n", "\n", @@ -101,11 +270,134 @@ "id": "ef0c31c5-9203-475b-abe1-5a0b53dc5e87", "metadata": {}, "source": [ - "## Import a circuit from qiskit (or other quantum computing library)\n", + "## Import a circuit from qiskit (or other SDK)\n", "\n", "Its possible to generate a circuit directly from a qiskit `QuantumCircuit` using the [qiskit_to_tk](https://tket.quantinuum.com/extensions/pytket-qiskit/api.html#pytket.extensions.qiskit.tk_to_qiskit) function." ] }, + { + "cell_type": "code", + "execution_count": 10, + "id": "23e753f2-3f6e-4a3d-a2e7-ece869561249", + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + " ┌───┐┌───┐ \n", + "q_0: ┤ H ├┤ X ├──■──\n", + " ├───┤└─┬─┘┌─┴─┐\n", + "q_1: ┤ H ├──■──┤ X ├\n", + " ├───┤ │ └───┘\n", + "q_2: ┤ H ├──■───────\n", + " └───┘ \n" + ] + } + ], + "source": [ + "from qiskit import QuantumCircuit\n", + "\n", + "qiskit_circ = QuantumCircuit(3)\n", + "qiskit_circ.h(range(3))\n", + "qiskit_circ.ccx(2, 1 ,0)\n", + "qiskit_circ.cx(0, 1)\n", + "print(qiskit_circ)" + ] + }, + { + "cell_type": "code", + "execution_count": 11, + "id": "5357e5d4-e0d8-4a59-9fe8-426067f411e0", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "\n", + "\n", + "\n", + "\n", + "\n", + "\n", + "
\n", + " \n", + "
\n", + "\n" + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + } + ], + "source": [ + "from pytket.extensions.qiskit import qiskit_to_tk\n", + "\n", + "tket_circ = qiskit_to_tk(qiskit_circ)\n", + "\n", + "render_circuit_jupyter(tket_circ)" + ] + }, { "cell_type": "markdown", "id": "7688630f-89e2-48c7-9d1e-53a9dedf088a", @@ -142,20 +434,107 @@ }, { "cell_type": "code", - "execution_count": null, + "execution_count": 6, "id": "42f4e1f6-2281-4780-bb11-7132c4be5313", "metadata": {}, - "outputs": [], + "outputs": [ + { + "data": { + "text/html": [ + "\n", + "\n", + "\n", + "\n", + "\n", + "\n", + "
\n", + " \n", + "
\n", + "\n" + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + } + ], "source": [ "render_circuit_jupyter(ghz_circ)" ] }, { "cell_type": "code", - "execution_count": null, + "execution_count": 7, "id": "868189e3-197c-427b-843e-6a991d1cfec9", "metadata": {}, - "outputs": [], + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Counter({(0, 0, 0): 520, (1, 1, 1): 504})\n" + ] + } + ], "source": [ "from pytket.extensions.qiskit import AerBackend\n", "\n", From df33f3ef93cef199067eba641c7aaad5431a85db Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Mon, 30 Oct 2023 10:58:26 +0000 Subject: [PATCH 30/51] don't execute notebook --- examples/Getting_started.ipynb | 377 ++------------------------------- 1 file changed, 14 insertions(+), 363 deletions(-) diff --git a/examples/Getting_started.ipynb b/examples/Getting_started.ipynb index 3dc02389..fb1ef8d5 100644 --- a/examples/Getting_started.ipynb +++ b/examples/Getting_started.ipynb @@ -21,21 +21,10 @@ }, { "cell_type": "code", - "execution_count": 1, + "execution_count": null, "id": "32a81360-0002-4c82-9a9e-a0f90851f34b", "metadata": {}, - "outputs": [ - { - "data": { - "text/plain": [ - "[H q[0]; CX q[0], q[1]; CX q[1], q[2]; Barrier q[0], q[1], q[2]; Measure q[0] --> c[0]; Measure q[1] --> c[1]; Measure q[2] --> c[2]; ]" - ] - }, - "execution_count": 1, - "metadata": {}, - "output_type": "execute_result" - } - ], + "outputs": [], "source": [ "from pytket import Circuit\n", "\n", @@ -57,89 +46,10 @@ }, { "cell_type": "code", - "execution_count": 2, + "execution_count": null, "id": "92df4f4d-487c-4c56-a072-9a1d2e3c07fc", "metadata": {}, - "outputs": [ - { - "data": { - "text/html": [ - "\n", - "\n", - "\n", - "\n", - "\n", - "\n", - "
\n", - " \n", - "
\n", - "\n" - ], - "text/plain": [ - "" - ] - }, - "metadata": {}, - "output_type": "display_data" - } - ], + "outputs": [], "source": [ "from pytket.circuit.display import render_circuit_jupyter\n", "\n", @@ -175,89 +85,10 @@ }, { "cell_type": "code", - "execution_count": 3, + "execution_count": null, "id": "1bf32877-f5e0-4b09-a12d-7aa8fa7d5d20", "metadata": {}, - "outputs": [ - { - "data": { - "text/html": [ - "\n", - "\n", - "\n", - "\n", - "\n", - "\n", - "
\n", - " \n", - "
\n", - "\n" - ], - "text/plain": [ - "" - ] - }, - "metadata": {}, - "output_type": "display_data" - } - ], + "outputs": [], "source": [ "from pytket.qasm import circuit_from_qasm\n", "\n", @@ -277,24 +108,10 @@ }, { "cell_type": "code", - "execution_count": 10, + "execution_count": null, "id": "23e753f2-3f6e-4a3d-a2e7-ece869561249", "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - " ┌───┐┌───┐ \n", - "q_0: ┤ H ├┤ X ├──■──\n", - " ├───┤└─┬─┘┌─┴─┐\n", - "q_1: ┤ H ├──■──┤ X ├\n", - " ├───┤ │ └───┘\n", - "q_2: ┤ H ├──■───────\n", - " └───┘ \n" - ] - } - ], + "outputs": [], "source": [ "from qiskit import QuantumCircuit\n", "\n", @@ -307,89 +124,10 @@ }, { "cell_type": "code", - "execution_count": 11, + "execution_count": null, "id": "5357e5d4-e0d8-4a59-9fe8-426067f411e0", "metadata": {}, - "outputs": [ - { - "data": { - "text/html": [ - "\n", - "\n", - "\n", - "\n", - "\n", - "\n", - "
\n", - " \n", - "
\n", - "\n" - ], - "text/plain": [ - "" - ] - }, - "metadata": {}, - "output_type": "display_data" - } - ], + "outputs": [], "source": [ "from pytket.extensions.qiskit import qiskit_to_tk\n", "\n", @@ -434,107 +172,20 @@ }, { "cell_type": "code", - "execution_count": 6, + "execution_count": null, "id": "42f4e1f6-2281-4780-bb11-7132c4be5313", "metadata": {}, - "outputs": [ - { - "data": { - "text/html": [ - "\n", - "\n", - "\n", - "\n", - "\n", - "\n", - "
\n", - " \n", - "
\n", - "\n" - ], - "text/plain": [ - "" - ] - }, - "metadata": {}, - "output_type": "display_data" - } - ], + "outputs": [], "source": [ "render_circuit_jupyter(ghz_circ)" ] }, { "cell_type": "code", - "execution_count": 7, + "execution_count": null, "id": "868189e3-197c-427b-843e-6a991d1cfec9", "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Counter({(0, 0, 0): 520, (1, 1, 1): 504})\n" - ] - } - ], + "outputs": [], "source": [ "from pytket.extensions.qiskit import AerBackend\n", "\n", From 07968228cd4d90f07cee53966326e8033c820838 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 2 Nov 2023 15:42:18 +0000 Subject: [PATCH 31/51] add images directory --- examples/{ => images}/IBMTokyoArc.png | Bin examples/{ => images}/IBMqx5Arc.png | Bin examples/{ => images}/phase_est.png | Bin examples/{ => images}/qft.png | Bin 4 files changed, 0 insertions(+), 0 deletions(-) rename examples/{ => images}/IBMTokyoArc.png (100%) rename examples/{ => images}/IBMqx5Arc.png (100%) rename examples/{ => images}/phase_est.png (100%) rename examples/{ => images}/qft.png (100%) diff --git a/examples/IBMTokyoArc.png b/examples/images/IBMTokyoArc.png similarity index 100% rename from examples/IBMTokyoArc.png rename to examples/images/IBMTokyoArc.png diff --git a/examples/IBMqx5Arc.png b/examples/images/IBMqx5Arc.png similarity index 100% rename from examples/IBMqx5Arc.png rename to examples/images/IBMqx5Arc.png diff --git a/examples/phase_est.png b/examples/images/phase_est.png similarity index 100% rename from examples/phase_est.png rename to examples/images/phase_est.png diff --git a/examples/qft.png b/examples/images/qft.png similarity index 100% rename from examples/qft.png rename to examples/images/qft.png From 6c82ed46631ea44efe22affaf620d4bfdc39a207 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 2 Nov 2023 15:42:42 +0000 Subject: [PATCH 32/51] clean up tket benchmarking example --- examples/tket_benchmarking.ipynb | 16 +--------------- 1 file changed, 1 insertion(+), 15 deletions(-) diff --git a/examples/tket_benchmarking.ipynb b/examples/tket_benchmarking.ipynb index 26aae780..29411e83 100644 --- a/examples/tket_benchmarking.ipynb +++ b/examples/tket_benchmarking.ipynb @@ -15,21 +15,7 @@ "\n", "The aim of this example is to show how to run the IBM benchmarking circuits through tket. You will need both `pytket` and `pytket_qiskit` installed from pip before running this turoial. You will also need `pandas` to capture the data.\n", "\n", - "The benchmarking circuits originated from https://github.com/iic-jku/ibm_qx_mapping/tree/master/examples, but there is a copy in pytket in the \"benchmarking\" folder. The initial circuits are written in QASM, meaning that they must be converted to tket's internal representation via Qiskit. Using this script we will compile these circuits through tket and then print a table to terminal containing analysis of the circuits post-compilation.\n", - "\n", - "First, begin by importing qiskit and pytket." - ] - }, - { - "cell_type": "code", - "execution_count": 1, - "metadata": { - "scrolled": true - }, - "outputs": [], - "source": [ - "import qiskit\n", - "import pytket" + "The benchmarking circuits originated from https://github.com/iic-jku/ibm_qx_mapping/tree/master/examples, but there is a copy in pytket in the \"benchmarking\" folder. The initial circuits are written in QASM, meaning that they must be converted to tket's internal representation via Qiskit. Using this script we will compile these circuits through tket and then print a table to terminal containing analysis of the circuits post-compilation." ] }, { From 170a90dd87d4e204ccc0ccb301ea16b5bcfc91f9 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 2 Nov 2023 15:45:55 +0000 Subject: [PATCH 33/51] fix QPE example --- examples/phase_estimation.ipynb | 2 +- examples/python/phase_estimation.py | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/examples/phase_estimation.ipynb b/examples/phase_estimation.ipynb index 80c0c9e2..488bf3ff 100644 --- a/examples/phase_estimation.ipynb +++ b/examples/phase_estimation.ipynb @@ -1 +1 @@ -{"cells":[{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["#!/usr/bin/env python\n","# coding: utf-8"]},{"cell_type":"markdown","metadata":{},"source":["# Quantum Phase Estimation using `pytket` Boxes\n","\n","When constructing circuits for quantum algorithms it is useful to think of higher level operations than just individual quantum gates.\n","\n","In `pytket` we can construct circuits using box structures which abstract away the complexity of the underlying circuit.\n","\n","This notebook is intended to complement the [boxes section](https://tket.quantinuum.com/user-manual/manual_circuit.html#boxes) of the user manual which introduces the different box types.\n","\n","To demonstrate boxes in `pytket` we will consider the Quantum Phase Estimation algorithm (QPE). This is an important subroutine in several quantum algorithms including Shor's algorithm and fault-tolerant approaches to quantum chemistry.\n","\n","## Overview of Phase Estimation\n","\n","The Quantum Phase Estimation algorithm can be used to estimate the eigenvalues of some unitary operator $U$ to some desired precision.\n","\n","The eigenvalues of $U$ lie on the unit circle, giving us the following eigenvalue equation\n","\n","$$\n","\\begin{equation}\n","U |\\psi \\rangle = e^{2 \\pi i \\theta} |\\psi\\rangle\\,, \\quad 0 \\leq \\theta \\leq 1\n","\\end{equation}\n","$$\n","\n","Here $|\\psi \\rangle$ is an eigenstate of the operator $U$. In phase estimation we estimate the eigenvalue $e^{2 \\pi i \\theta}$ by approximating $\\theta$.\n","\n","\n","The circuit for Quantum phase estimation is itself composed of several subroutines which we can realise as boxes.\n","\n","![](phase_est.png \"Quantum Phase Estimation Circuit\")"]},{"cell_type":"markdown","metadata":{},"source":["QPE is generally split up into three stages\n","\n","1. Firstly we prepare an initial state in one register. In parallel we prepare a uniform superposition state using Hadamard gates on some ancilla qubits. The number of ancilla qubits determines how precisely we can estimate the phase $\\theta$.\n","\n","2. Secondly we apply successive controlled $U$ gates. This has the effect of \"kicking back\" phases onto the ancilla qubits according to the eigenvalue equation above.\n","\n","3. Finally we apply the inverse Quantum Fourier Transform (QFT). This essentially plays the role of destructive interference, suppressing amplitudes from \"undesirable states\" and hopefully allowing us to measure a single outcome (or a small number of outcomes) with high probability.\n","\n","\n","There is some subtlety around the first point. The initial state used can be an exact eigenstate of $U$ however this may be difficult to prepare if we don't know the eigenvalues of $U$ in advance. Alternatively we could use an initial state that is a linear combination of eigenstates, as the phase estimation will project into the eigenspace of $U$."]},{"cell_type":"markdown","metadata":{},"source":["We also assume that we can implement $U$ with a quantum circuit. In chemistry applications $U$ could be of the form $U=e^{-iHt}$ where $H$ is the Hamiltonian of some system of interest. In the cannonical algorithm, the number of controlled unitaries we apply scales exponentially with the number of ancilla qubits. This allows more precision at the expense of a larger quantum circuit."]},{"cell_type":"markdown","metadata":{},"source":["## The Quantum Fourier Transform"]},{"cell_type":"markdown","metadata":{},"source":["Before considering the other parts of the QPE algorithm, lets focus on the Quantum Fourier Transform (QFT) subroutine.\n","\n","Mathematically, the QFT has the following action.\n","\n","\\begin{equation}\n","QFT : |j\\rangle\\ \\longmapsto \\sum_{k=0}^{N - 1} e^{2 \\pi ijk/N}|k\\rangle, \\quad N= 2^k\n","\\end{equation}\n","\n","This is essentially the Discrete Fourier transform except the input is a quantum state $|j\\rangle$.\n","\n","It is well known that the QFT can be implemented efficiently with a quantum circuit\n","\n","We can build the circuit for the $n$ qubit QFT using $n$ Hadamard gates $\\frac{n}{2}$ swap gates and $\\frac{n(n-1)}{2}$ controlled unitary rotations $\\text{CU1}$.\n","\n","$$\n"," \\begin{equation}\n"," CU1(\\phi) =\n"," \\begin{pmatrix}\n"," I & 0 \\\\\n"," 0 & U1(\\phi)\n"," \\end{pmatrix}\n"," \\,, \\quad\n","U1(\\phi) =\n"," \\begin{pmatrix}\n"," 1 & 0 \\\\\n"," 0 & e^{i \\phi}\n"," \\end{pmatrix}\n"," \\end{equation}\n","$$\n","\n","The circuit for the Quantum Fourier transform on three qubits is the following\n","\n","![](qft.png \"QFT Circuit\")\n","\n","We can build this circuit in `pytket` by adding gate operations manually:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import Circuit\n","from pytket.circuit.display import render_circuit_jupyter"]},{"cell_type":"markdown","metadata":{},"source":["lets build the QFT for three qubits"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft3_circ = Circuit(3)\n","qft3_circ.H(0)\n","qft3_circ.CU1(0.5, 1, 0)\n","qft3_circ.CU1(0.25, 2, 0)\n","qft3_circ.H(1)\n","qft3_circ.CU1(0.5, 2, 1)\n","qft3_circ.H(2)\n","qft3_circ.SWAP(0, 2)\n","render_circuit_jupyter(qft3_circ)"]},{"cell_type":"markdown","metadata":{},"source":["We can generalise the quantum Fourier transform to $n$ qubits by iterating over the qubits as follows"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def build_qft_circuit(n_qubits: int) -> Circuit:\n"," circ = Circuit(n_qubits, name=\"QFT\")\n"," for i in range(n_qubits):\n"," circ.H(i)\n"," for j in range(i + 1, n_qubits):\n"," circ.CU1(1 / 2 ** (j - i), j, i)\n"," for k in range(0, n_qubits // 2):\n"," circ.SWAP(k, n_qubits - k - 1)\n"," return circ"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft4_circ: Circuit = build_qft_circuit(4)\n","render_circuit_jupyter(qft4_circ)"]},{"cell_type":"markdown","metadata":{},"source":["Now that we have the generalised circuit we can wrap it up in a `CircBox` which can then be added to another circuit as a subroutine."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import CircBox"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft4_box: CircBox = CircBox(qft4_circ)\n","qft_circ = Circuit(4).add_gate(qft4_box, [0, 1, 2, 3])\n","render_circuit_jupyter(qft_circ)"]},{"cell_type":"markdown","metadata":{},"source":["Note how the `CircBox` inherits the name `QFT` from the underlying circuit."]},{"cell_type":"markdown","metadata":{},"source":["Recall that in our phase estimation algorithm we need to use the inverse QFT.\n","\n","$$\n","\\begin{equation}\n","\\text{QFT}^† : \\sum_{k=0}^{N - 1} e^{2 \\pi ijk/N}|k\\rangle \\longmapsto |j\\rangle\\,, \\quad N= 2^k\n","\\end{equation}\n","$$\n","\n","\n","Now that we have the QFT circuit we can obtain the inverse by using `CircBox.dagger`. We can also verify that this is correct by inspecting the circuit inside with `CircBox.get_circuit()`."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["inv_qft4_box = qft4_box.dagger\n","render_circuit_jupyter(inv_qft4_box.get_circuit())"]},{"cell_type":"markdown","metadata":{},"source":["## The Controlled Unitary Operations"]},{"cell_type":"markdown","metadata":{},"source":["In the phase estimation algorithm we repeatedly perform controlled unitary operations. In the canonical variant, the number of controlled unitaries will be $2^m - 1$ where $m$ is the number of measurement qubits."]},{"cell_type":"markdown","metadata":{},"source":["The form of $U$ will vary depending on the application. For chemistry or condensed matter physics $U$ typically be the time evolution operator $U(t) = e^{- i H t}$ where $H$ is the problem Hamiltonian."]},{"cell_type":"markdown","metadata":{},"source":["Suppose that we had the following decomposition for $H$ in terms of Pauli strings $P_j$ and complex coefficients $\\alpha_j$.\n","\n","\\begin{equation}\n","H = \\sum_j \\alpha_j P_j\\,, \\quad \\, P_j \\in \\{I, X, Y, Z\\}^{\\otimes n}\n","\\end{equation}\n","\n","Here Pauli strings refers to tensor products of Pauli operators. These strings form an orthonormal basis for $2^n \\times 2^n$ matrices."]},{"cell_type":"markdown","metadata":{},"source":["If we have a Hamiltonian in the form above, we can then implement $U(t)$ as a sequence of Pauli gadget circuits. We can do this with the [PauliExpBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.PauliExpBox) construct in pytket. For more on `PauliExpBox` see the [user manual](https://tket.quantinuum.com/user-manual/manual_circuit.html#pauli-exponential-boxes)."]},{"cell_type":"markdown","metadata":{},"source":["Once we have a circuit to implement our time evolution operator $U(t)$, we can construct the controlled $U(t)$ operations using [QControlBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.QControlBox). If our base unitary is a sequence of `PauliExpBox`(es) then there is some structure we can exploit to simplify our circuit. See this [blog post](https://tket.quantinuum.com/tket-blog/posts/controlled_gates/) on [ConjugationBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.ConjugationBox) for more."]},{"cell_type":"markdown","metadata":{},"source":["In what follows, we will just construct a simplified instance of QPE where the controlled unitaries are just $\\text{CU1}$ gates."]},{"cell_type":"markdown","metadata":{},"source":["## Putting it all together"]},{"cell_type":"markdown","metadata":{},"source":["We can now define a function to build our entire QPE circuit. We can make this function take a state preparation circuit and a unitary circuit as input as well. The function also has the number of measurement qubits as input which will determine the precision of our phase estimate."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import QControlBox"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def build_phase_est_circuit(\n"," n_measurement_qubits: int, state_prep_circuit: Circuit, unitary_circuit: Circuit\n",") -> Circuit:\n"," qpe_circ: Circuit = Circuit()\n"," n_state_prep_qubits = state_prep_circuit.n_qubits\n"," measurement_register = qpe_circ.add_q_register(\"m\", n_measurement_qubits)\n"," state_prep_register = qpe_circ.add_q_register(\"p\", n_state_prep_qubits)\n"," qpe_circ.add_circuit(state_prep_circuit, list(state_prep_register))\n","\n"," # Create a controlled unitary with a single control qubit\n"," unitary_circuit.name = \"U\"\n"," controlled_u_gate = QControlBox(CircBox(unitary_circuit), 1)\n","\n"," # Add Hadamard gates to every qubit in the measurement register\n"," for m_qubit in measurement_register:\n"," qpe_circ.H(m_qubit)\n","\n"," # Add all (2**n_measurement_qubits - 1) of the controlled unitaries sequentially\n"," for m_qubit in range(n_measurement_qubits):\n"," control_index = n_measurement_qubits - m_qubit - 1\n"," control_qubit = [measurement_register[control_index]]\n"," for _ in range(2**m_qubit):\n"," qpe_circ.add_qcontrolbox(\n"," controlled_u_gate, control_qubit + list(state_prep_register)\n"," )\n","\n"," # Finally, append the inverse qft and measure the qubits\n"," qft_box = CircBox(build_qft_circuit(n_measurement_qubits))\n"," inverse_qft_box = qft_box.dagger\n"," qpe_circ.add_circbox(inverse_qft_box, list(measurement_register))\n"," qpe_circ.measure_register(measurement_register, \"c\")\n"," return qpe_circ"]},{"cell_type":"markdown","metadata":{},"source":["## Phase Estimation with a Trivial Eigenstate\n","\n","Lets test our circuit construction by preparing a trivial $|1\\rangle$ eigenstate of the $\\text{U1}$ gate. We can then see if our phase estimation circuit returns the expected eigenvalue."]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","U1(\\phi)|1\\rangle = e^{i\\phi} = e^{2 \\pi i \\theta} \\implies \\theta = \\frac{\\phi}{2}\n","\\end{equation}\n","$$\n","\n","So we expect that our ideal phase $\\theta$ will be half the input angle $\\phi$ to our $U1$ gate."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["prep_circuit = Circuit(1).X(0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["input_angle = 0.73 # angle as number of half turns"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["unitary_circuit = Circuit(1).U1(input_angle, 0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qpe_circ_trivial = build_phase_est_circuit(\n"," 4, state_prep_circuit=prep_circuit, unitary_circuit=unitary_circuit\n",")"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["render_circuit_jupyter(qpe_circ_trivial)"]},{"cell_type":"markdown","metadata":{},"source":["Lets use the noiseless `AerBackend` simulator to run our phase estimation circuit."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.extensions.qiskit import AerBackend"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["backend = AerBackend()"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["compiled_circ = backend.get_compiled_circuit(qpe_circ_trivial)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_shots = 1000\n","result = backend.run_circuit(compiled_circ, n_shots)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(result.get_counts())"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.backends.backendresult import BackendResult\n","import matplotlib.pyplot as plt"]},{"cell_type":"markdown","metadata":{},"source":["plotting function for QPE Notebook"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def plot_qpe_results(\n"," sim_result: BackendResult,\n"," n_strings: int = 4,\n"," dark_mode: bool = False,\n"," y_limit: int = 1000,\n",") -> None:\n"," \"\"\"\n"," Plots results in a barchart given a BackendResult. the number of stings displayed\n"," can be specified with the n_strings argument.\n"," \"\"\"\n"," counts_dict = sim_result.get_counts()\n"," sorted_shots = counts_dict.most_common()\n"," n_most_common_strings = sorted_shots[:n_strings]\n"," x_axis_values = [str(entry[0]) for entry in n_most_common_strings] # basis states\n"," y_axis_values = [entry[1] for entry in n_most_common_strings] # counts\n"," if dark_mode:\n"," plt.style.use(\"dark_background\")\n"," fig = plt.figure()\n"," ax = fig.add_axes((0, 0, 0.75, 0.5))\n"," color_list = [\"orange\"] * (len(x_axis_values))\n"," ax.bar(\n"," x=x_axis_values,\n"," height=y_axis_values,\n"," color=color_list,\n"," )\n"," ax.set_title(label=\"Results\")\n"," plt.ylim([0, y_limit])\n"," plt.xlabel(\"Basis State\")\n"," plt.ylabel(\"Number of Shots\")\n"," plt.xticks(rotation=90)\n"," plt.show()"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plot_qpe_results(result, y_limit=int(1.2 * n_shots))"]},{"cell_type":"markdown","metadata":{},"source":["As expected we see one outcome with high probability. Lets now extract our approximation of $\\theta$ from our output bitstrings.\n","\n","suppose the $j$ is an integer representation of our most commonly measured bitstring."]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","\\theta_{estimate} = \\frac{j}{N}\n","\\end{equation}\n","$$"]},{"cell_type":"markdown","metadata":{},"source":["Here $N = 2 ^n$ where $n$ is the number of measurement qubits."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.backends.backendresult import BackendResult"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def single_phase_from_backendresult(result: BackendResult) -> float:\n"," # Extract most common measurement outcome\n"," basis_state = result.get_counts().most_common()[0][0]\n"," bitstring = \"\".join([str(bit) for bit in basis_state])\n"," integer = int(bitstring, 2)\n","\n"," # Calculate theta estimate\n"," return integer / (2 ** len(bitstring))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["theta = single_phase_from_backendresult(result)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(theta)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(input_angle / 2)"]},{"cell_type":"markdown","metadata":{},"source":["Our output is close to half our input angle $\\phi$ as expected. Lets calculate our error."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["error = round(abs(input_angle - (2 * theta)), 3)\n","print(error)"]},{"cell_type":"markdown","metadata":{},"source":["## Suggestions for further reading\n","\n","In this notebook we have shown the canonical variant of quantum phase estimation. There are several other variants.\n","\n","Quantinuum paper on Bayesian phase estimation -> https://arxiv.org/pdf/2306.16608.pdf\n","Blog post on `ConjugationBox` -> https://tket.quantinuum.com/tket-blog/posts/controlled_gates/ - efficient circuits for controlled Pauli gadgets.\n","\n","As mentioned quantum phase estimation is a subroutine in Shor's algorithm. Read more about how phase estimation is used in period finding."]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} +{"cells":[{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["#!/usr/bin/env python\n","# coding: utf-8"]},{"cell_type":"markdown","metadata":{},"source":["# Quantum Phase Estimation using `pytket` Boxes\n","\n","When constructing circuits for quantum algorithms it is useful to think of higher level operations than just individual quantum gates.\n","\n","In `pytket` we can construct circuits using box structures which abstract away the complexity of the underlying circuit.\n","\n","This notebook is intended to complement the [boxes section](https://tket.quantinuum.com/user-manual/manual_circuit.html#boxes) of the user manual which introduces the different box types.\n","\n","To demonstrate boxes in `pytket` we will consider the Quantum Phase Estimation algorithm (QPE). This is an important subroutine in several quantum algorithms including Shor's algorithm and fault-tolerant approaches to quantum chemistry.\n","\n","## Overview of Phase Estimation\n","\n","The Quantum Phase Estimation algorithm can be used to estimate the eigenvalues of some unitary operator $U$ to some desired precision.\n","\n","The eigenvalues of $U$ lie on the unit circle, giving us the following eigenvalue equation\n","\n","$$\n","\\begin{equation}\n","U |\\psi \\rangle = e^{2 \\pi i \\theta} |\\psi\\rangle\\,, \\quad 0 \\leq \\theta \\leq 1\n","\\end{equation}\n","$$\n","\n","Here $|\\psi \\rangle$ is an eigenstate of the operator $U$. In phase estimation we estimate the eigenvalue $e^{2 \\pi i \\theta}$ by approximating $\\theta$.\n","\n","\n","The circuit for Quantum phase estimation is itself composed of several subroutines which we can realise as boxes.\n","\n","![](images/phase_est.png \"Quantum Phase Estimation Circuit\")"]},{"cell_type":"markdown","metadata":{},"source":["QPE is generally split up into three stages\n","\n","1. Firstly we prepare an initial state in one register. In parallel we prepare a uniform superposition state using Hadamard gates on some ancilla qubits. The number of ancilla qubits determines how precisely we can estimate the phase $\\theta$.\n","\n","2. Secondly we apply successive controlled $U$ gates. This has the effect of \"kicking back\" phases onto the ancilla qubits according to the eigenvalue equation above.\n","\n","3. Finally we apply the inverse Quantum Fourier Transform (QFT). This essentially plays the role of destructive interference, suppressing amplitudes from \"undesirable states\" and hopefully allowing us to measure a single outcome (or a small number of outcomes) with high probability.\n","\n","\n","There is some subtlety around the first point. The initial state used can be an exact eigenstate of $U$ however this may be difficult to prepare if we don't know the eigenvalues of $U$ in advance. Alternatively we could use an initial state that is a linear combination of eigenstates, as the phase estimation will project into the eigenspace of $U$."]},{"cell_type":"markdown","metadata":{},"source":["We also assume that we can implement $U$ with a quantum circuit. In chemistry applications $U$ could be of the form $U=e^{-iHt}$ where $H$ is the Hamiltonian of some system of interest. In the cannonical algorithm, the number of controlled unitaries we apply scales exponentially with the number of ancilla qubits. This allows more precision at the expense of a larger quantum circuit."]},{"cell_type":"markdown","metadata":{},"source":["## The Quantum Fourier Transform"]},{"cell_type":"markdown","metadata":{},"source":["Before considering the other parts of the QPE algorithm, lets focus on the Quantum Fourier Transform (QFT) subroutine.\n","\n","Mathematically, the QFT has the following action.\n","\n","\\begin{equation}\n","QFT : |j\\rangle\\ \\longmapsto \\sum_{k=0}^{N - 1} e^{2 \\pi ijk/N}|k\\rangle, \\quad N= 2^k\n","\\end{equation}\n","\n","This is essentially the Discrete Fourier transform except the input is a quantum state $|j\\rangle$.\n","\n","It is well known that the QFT can be implemented efficiently with a quantum circuit\n","\n","We can build the circuit for the $n$ qubit QFT using $n$ Hadamard gates $\\frac{n}{2}$ swap gates and $\\frac{n(n-1)}{2}$ controlled unitary rotations $\\text{CU1}$.\n","\n","$$\n"," \\begin{equation}\n"," CU1(\\phi) =\n"," \\begin{pmatrix}\n"," I & 0 \\\\\n"," 0 & U1(\\phi)\n"," \\end{pmatrix}\n"," \\,, \\quad\n","U1(\\phi) =\n"," \\begin{pmatrix}\n"," 1 & 0 \\\\\n"," 0 & e^{i \\phi}\n"," \\end{pmatrix}\n"," \\end{equation}\n","$$\n","\n","The circuit for the Quantum Fourier transform on three qubits is the following\n","\n","![](images/qft.png \"QFT Circuit\")\n","\n","We can build this circuit in `pytket` by adding gate operations manually:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import Circuit\n","from pytket.circuit.display import render_circuit_jupyter"]},{"cell_type":"markdown","metadata":{},"source":["lets build the QFT for three qubits"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft3_circ = Circuit(3)\n","qft3_circ.H(0)\n","qft3_circ.CU1(0.5, 1, 0)\n","qft3_circ.CU1(0.25, 2, 0)\n","qft3_circ.H(1)\n","qft3_circ.CU1(0.5, 2, 1)\n","qft3_circ.H(2)\n","qft3_circ.SWAP(0, 2)\n","render_circuit_jupyter(qft3_circ)"]},{"cell_type":"markdown","metadata":{},"source":["We can generalise the quantum Fourier transform to $n$ qubits by iterating over the qubits as follows"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def build_qft_circuit(n_qubits: int) -> Circuit:\n"," circ = Circuit(n_qubits, name=\"QFT\")\n"," for i in range(n_qubits):\n"," circ.H(i)\n"," for j in range(i + 1, n_qubits):\n"," circ.CU1(1 / 2 ** (j - i), j, i)\n"," for k in range(0, n_qubits // 2):\n"," circ.SWAP(k, n_qubits - k - 1)\n"," return circ"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft4_circ: Circuit = build_qft_circuit(4)\n","render_circuit_jupyter(qft4_circ)"]},{"cell_type":"markdown","metadata":{},"source":["Now that we have the generalised circuit we can wrap it up in a `CircBox` which can then be added to another circuit as a subroutine."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import CircBox"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft4_box: CircBox = CircBox(qft4_circ)\n","qft_circ = Circuit(4).add_gate(qft4_box, [0, 1, 2, 3])\n","render_circuit_jupyter(qft_circ)"]},{"cell_type":"markdown","metadata":{},"source":["Note how the `CircBox` inherits the name `QFT` from the underlying circuit."]},{"cell_type":"markdown","metadata":{},"source":["Recall that in our phase estimation algorithm we need to use the inverse QFT.\n","\n","$$\n","\\begin{equation}\n","\\text{QFT}^† : \\sum_{k=0}^{N - 1} e^{2 \\pi ijk/N}|k\\rangle \\longmapsto |j\\rangle\\,, \\quad N= 2^k\n","\\end{equation}\n","$$\n","\n","\n","Now that we have the QFT circuit we can obtain the inverse by using `CircBox.dagger`. We can also verify that this is correct by inspecting the circuit inside with `CircBox.get_circuit()`."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["inv_qft4_box = qft4_box.dagger\n","render_circuit_jupyter(inv_qft4_box.get_circuit())"]},{"cell_type":"markdown","metadata":{},"source":["## The Controlled Unitary Operations"]},{"cell_type":"markdown","metadata":{},"source":["In the phase estimation algorithm we repeatedly perform controlled unitary operations. In the canonical variant, the number of controlled unitaries will be $2^m - 1$ where $m$ is the number of measurement qubits."]},{"cell_type":"markdown","metadata":{},"source":["The form of $U$ will vary depending on the application. For chemistry or condensed matter physics $U$ typically be the time evolution operator $U(t) = e^{- i H t}$ where $H$ is the problem Hamiltonian."]},{"cell_type":"markdown","metadata":{},"source":["Suppose that we had the following decomposition for $H$ in terms of Pauli strings $P_j$ and complex coefficients $\\alpha_j$.\n","\n","\\begin{equation}\n","H = \\sum_j \\alpha_j P_j\\,, \\quad \\, P_j \\in \\{I, X, Y, Z\\}^{\\otimes n}\n","\\end{equation}\n","\n","Here Pauli strings refers to tensor products of Pauli operators. These strings form an orthonormal basis for $2^n \\times 2^n$ matrices."]},{"cell_type":"markdown","metadata":{},"source":["If we have a Hamiltonian in the form above, we can then implement $U(t)$ as a sequence of Pauli gadget circuits. We can do this with the [PauliExpBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.PauliExpBox) construct in pytket. For more on `PauliExpBox` see the [user manual](https://tket.quantinuum.com/user-manual/manual_circuit.html#pauli-exponential-boxes)."]},{"cell_type":"markdown","metadata":{},"source":["Once we have a circuit to implement our time evolution operator $U(t)$, we can construct the controlled $U(t)$ operations using [QControlBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.QControlBox). If our base unitary is a sequence of `PauliExpBox`(es) then there is some structure we can exploit to simplify our circuit. See this [blog post](https://tket.quantinuum.com/tket-blog/posts/controlled_gates/) on [ConjugationBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.ConjugationBox) for more."]},{"cell_type":"markdown","metadata":{},"source":["In what follows, we will just construct a simplified instance of QPE where the controlled unitaries are just $\\text{CU1}$ gates."]},{"cell_type":"markdown","metadata":{},"source":["## Putting it all together"]},{"cell_type":"markdown","metadata":{},"source":["We can now define a function to build our entire QPE circuit. We can make this function take a state preparation circuit and a unitary circuit as input as well. The function also has the number of measurement qubits as input which will determine the precision of our phase estimate."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import QControlBox"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def build_phase_est_circuit(\n"," n_measurement_qubits: int, state_prep_circuit: Circuit, unitary_circuit: Circuit\n",") -> Circuit:\n"," qpe_circ: Circuit = Circuit()\n"," n_state_prep_qubits = state_prep_circuit.n_qubits\n"," measurement_register = qpe_circ.add_q_register(\"m\", n_measurement_qubits)\n"," state_prep_register = qpe_circ.add_q_register(\"p\", n_state_prep_qubits)\n"," qpe_circ.add_circuit(state_prep_circuit, list(state_prep_register))\n","\n"," # Create a controlled unitary with a single control qubit\n"," unitary_circuit.name = \"U\"\n"," controlled_u_gate = QControlBox(CircBox(unitary_circuit), 1)\n","\n"," # Add Hadamard gates to every qubit in the measurement register\n"," for m_qubit in measurement_register:\n"," qpe_circ.H(m_qubit)\n","\n"," # Add all (2**n_measurement_qubits - 1) of the controlled unitaries sequentially\n"," for m_qubit in range(n_measurement_qubits):\n"," control_index = n_measurement_qubits - m_qubit - 1\n"," control_qubit = [measurement_register[control_index]]\n"," for _ in range(2**m_qubit):\n"," qpe_circ.add_qcontrolbox(\n"," controlled_u_gate, control_qubit + list(state_prep_register)\n"," )\n","\n"," # Finally, append the inverse qft and measure the qubits\n"," qft_box = CircBox(build_qft_circuit(n_measurement_qubits))\n"," inverse_qft_box = qft_box.dagger\n"," qpe_circ.add_circbox(inverse_qft_box, list(measurement_register))\n"," qpe_circ.measure_register(measurement_register, \"c\")\n"," return qpe_circ"]},{"cell_type":"markdown","metadata":{},"source":["## Phase Estimation with a Trivial Eigenstate\n","\n","Lets test our circuit construction by preparing a trivial $|1\\rangle$ eigenstate of the $\\text{U1}$ gate. We can then see if our phase estimation circuit returns the expected eigenvalue."]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","U1(\\phi)|1\\rangle = e^{i\\phi} = e^{2 \\pi i \\theta} \\implies \\theta = \\frac{\\phi}{2}\n","\\end{equation}\n","$$\n","\n","So we expect that our ideal phase $\\theta$ will be half the input angle $\\phi$ to our $U1$ gate."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["prep_circuit = Circuit(1).X(0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["input_angle = 0.73 # angle as number of half turns"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["unitary_circuit = Circuit(1).U1(input_angle, 0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qpe_circ_trivial = build_phase_est_circuit(\n"," 4, state_prep_circuit=prep_circuit, unitary_circuit=unitary_circuit\n",")"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["render_circuit_jupyter(qpe_circ_trivial)"]},{"cell_type":"markdown","metadata":{},"source":["Lets use the noiseless `AerBackend` simulator to run our phase estimation circuit."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.extensions.qiskit import AerBackend"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["backend = AerBackend()"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["compiled_circ = backend.get_compiled_circuit(qpe_circ_trivial)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_shots = 1000\n","result = backend.run_circuit(compiled_circ, n_shots)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(result.get_counts())"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.backends.backendresult import BackendResult\n","import matplotlib.pyplot as plt"]},{"cell_type":"markdown","metadata":{},"source":["plotting function for QPE Notebook"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def plot_qpe_results(\n"," sim_result: BackendResult,\n"," n_strings: int = 4,\n"," dark_mode: bool = False,\n"," y_limit: int = 1000,\n",") -> None:\n"," \"\"\"\n"," Plots results in a barchart given a BackendResult. the number of stings displayed\n"," can be specified with the n_strings argument.\n"," \"\"\"\n"," counts_dict = sim_result.get_counts()\n"," sorted_shots = counts_dict.most_common()\n"," n_most_common_strings = sorted_shots[:n_strings]\n"," x_axis_values = [str(entry[0]) for entry in n_most_common_strings] # basis states\n"," y_axis_values = [entry[1] for entry in n_most_common_strings] # counts\n"," if dark_mode:\n"," plt.style.use(\"dark_background\")\n"," fig = plt.figure()\n"," ax = fig.add_axes((0, 0, 0.75, 0.5))\n"," color_list = [\"orange\"] * (len(x_axis_values))\n"," ax.bar(\n"," x=x_axis_values,\n"," height=y_axis_values,\n"," color=color_list,\n"," )\n"," ax.set_title(label=\"Results\")\n"," plt.ylim([0, y_limit])\n"," plt.xlabel(\"Basis State\")\n"," plt.ylabel(\"Number of Shots\")\n"," plt.xticks(rotation=90)\n"," plt.show()"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plot_qpe_results(result, y_limit=int(1.2 * n_shots))"]},{"cell_type":"markdown","metadata":{},"source":["As expected we see one outcome with high probability. Lets now extract our approximation of $\\theta$ from our output bitstrings.\n","\n","suppose the $j$ is an integer representation of our most commonly measured bitstring."]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","\\theta_{estimate} = \\frac{j}{N}\n","\\end{equation}\n","$$"]},{"cell_type":"markdown","metadata":{},"source":["Here $N = 2 ^n$ where $n$ is the number of measurement qubits."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.backends.backendresult import BackendResult"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def single_phase_from_backendresult(result: BackendResult) -> float:\n"," # Extract most common measurement outcome\n"," basis_state = result.get_counts().most_common()[0][0]\n"," bitstring = \"\".join([str(bit) for bit in basis_state])\n"," integer = int(bitstring, 2)\n","\n"," # Calculate theta estimate\n"," return integer / (2 ** len(bitstring))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["theta = single_phase_from_backendresult(result)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(theta)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(input_angle / 2)"]},{"cell_type":"markdown","metadata":{},"source":["Our output is close to half our input angle $\\phi$ as expected. Lets calculate our error."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["error = round(abs(input_angle - (2 * theta)), 3)\n","print(error)"]},{"cell_type":"markdown","metadata":{},"source":["## Suggestions for further reading\n","\n","In this notebook we have shown the canonical variant of quantum phase estimation. There are several other variants.\n","\n","Quantinuum paper on Bayesian phase estimation -> https://arxiv.org/pdf/2306.16608.pdf\n","Blog post on `ConjugationBox` -> https://tket.quantinuum.com/tket-blog/posts/controlled_gates/ - efficient circuits for controlled Pauli gadgets.\n","\n","As mentioned quantum phase estimation is a subroutine in Shor's algorithm. Read more about how phase estimation is used in period finding."]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} diff --git a/examples/python/phase_estimation.py b/examples/python/phase_estimation.py index 7e9fe0ee..1eaf3937 100644 --- a/examples/python/phase_estimation.py +++ b/examples/python/phase_estimation.py @@ -28,7 +28,7 @@ # # The circuit for Quantum phase estimation is itself composed of several subroutines which we can realise as boxes. # -# ![](phase_est.png "Quantum Phase Estimation Circuit") +# ![](images/phase_est.png "Quantum Phase Estimation Circuit") # QPE is generally split up into three stages # @@ -77,7 +77,7 @@ # # The circuit for the Quantum Fourier transform on three qubits is the following # -# ![](qft.png "QFT Circuit") +# ![](images/qft.png "QFT Circuit") # # We can build this circuit in `pytket` by adding gate operations manually: From b7fc27a8a99c8c03e5bd84abdc7b2aa93a8a6af2 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Tue, 7 Nov 2023 11:21:38 +0000 Subject: [PATCH 34/51] clean up QPE example --- examples/phase_estimation.ipynb | 2 +- examples/python/phase_estimation.py | 15 +++++++-------- 2 files changed, 8 insertions(+), 9 deletions(-) diff --git a/examples/phase_estimation.ipynb b/examples/phase_estimation.ipynb index 488bf3ff..f68b02f8 100644 --- a/examples/phase_estimation.ipynb +++ b/examples/phase_estimation.ipynb @@ -1 +1 @@ -{"cells":[{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["#!/usr/bin/env python\n","# coding: utf-8"]},{"cell_type":"markdown","metadata":{},"source":["# Quantum Phase Estimation using `pytket` Boxes\n","\n","When constructing circuits for quantum algorithms it is useful to think of higher level operations than just individual quantum gates.\n","\n","In `pytket` we can construct circuits using box structures which abstract away the complexity of the underlying circuit.\n","\n","This notebook is intended to complement the [boxes section](https://tket.quantinuum.com/user-manual/manual_circuit.html#boxes) of the user manual which introduces the different box types.\n","\n","To demonstrate boxes in `pytket` we will consider the Quantum Phase Estimation algorithm (QPE). This is an important subroutine in several quantum algorithms including Shor's algorithm and fault-tolerant approaches to quantum chemistry.\n","\n","## Overview of Phase Estimation\n","\n","The Quantum Phase Estimation algorithm can be used to estimate the eigenvalues of some unitary operator $U$ to some desired precision.\n","\n","The eigenvalues of $U$ lie on the unit circle, giving us the following eigenvalue equation\n","\n","$$\n","\\begin{equation}\n","U |\\psi \\rangle = e^{2 \\pi i \\theta} |\\psi\\rangle\\,, \\quad 0 \\leq \\theta \\leq 1\n","\\end{equation}\n","$$\n","\n","Here $|\\psi \\rangle$ is an eigenstate of the operator $U$. In phase estimation we estimate the eigenvalue $e^{2 \\pi i \\theta}$ by approximating $\\theta$.\n","\n","\n","The circuit for Quantum phase estimation is itself composed of several subroutines which we can realise as boxes.\n","\n","![](images/phase_est.png \"Quantum Phase Estimation Circuit\")"]},{"cell_type":"markdown","metadata":{},"source":["QPE is generally split up into three stages\n","\n","1. Firstly we prepare an initial state in one register. In parallel we prepare a uniform superposition state using Hadamard gates on some ancilla qubits. The number of ancilla qubits determines how precisely we can estimate the phase $\\theta$.\n","\n","2. Secondly we apply successive controlled $U$ gates. This has the effect of \"kicking back\" phases onto the ancilla qubits according to the eigenvalue equation above.\n","\n","3. Finally we apply the inverse Quantum Fourier Transform (QFT). This essentially plays the role of destructive interference, suppressing amplitudes from \"undesirable states\" and hopefully allowing us to measure a single outcome (or a small number of outcomes) with high probability.\n","\n","\n","There is some subtlety around the first point. The initial state used can be an exact eigenstate of $U$ however this may be difficult to prepare if we don't know the eigenvalues of $U$ in advance. Alternatively we could use an initial state that is a linear combination of eigenstates, as the phase estimation will project into the eigenspace of $U$."]},{"cell_type":"markdown","metadata":{},"source":["We also assume that we can implement $U$ with a quantum circuit. In chemistry applications $U$ could be of the form $U=e^{-iHt}$ where $H$ is the Hamiltonian of some system of interest. In the cannonical algorithm, the number of controlled unitaries we apply scales exponentially with the number of ancilla qubits. This allows more precision at the expense of a larger quantum circuit."]},{"cell_type":"markdown","metadata":{},"source":["## The Quantum Fourier Transform"]},{"cell_type":"markdown","metadata":{},"source":["Before considering the other parts of the QPE algorithm, lets focus on the Quantum Fourier Transform (QFT) subroutine.\n","\n","Mathematically, the QFT has the following action.\n","\n","\\begin{equation}\n","QFT : |j\\rangle\\ \\longmapsto \\sum_{k=0}^{N - 1} e^{2 \\pi ijk/N}|k\\rangle, \\quad N= 2^k\n","\\end{equation}\n","\n","This is essentially the Discrete Fourier transform except the input is a quantum state $|j\\rangle$.\n","\n","It is well known that the QFT can be implemented efficiently with a quantum circuit\n","\n","We can build the circuit for the $n$ qubit QFT using $n$ Hadamard gates $\\frac{n}{2}$ swap gates and $\\frac{n(n-1)}{2}$ controlled unitary rotations $\\text{CU1}$.\n","\n","$$\n"," \\begin{equation}\n"," CU1(\\phi) =\n"," \\begin{pmatrix}\n"," I & 0 \\\\\n"," 0 & U1(\\phi)\n"," \\end{pmatrix}\n"," \\,, \\quad\n","U1(\\phi) =\n"," \\begin{pmatrix}\n"," 1 & 0 \\\\\n"," 0 & e^{i \\phi}\n"," \\end{pmatrix}\n"," \\end{equation}\n","$$\n","\n","The circuit for the Quantum Fourier transform on three qubits is the following\n","\n","![](images/qft.png \"QFT Circuit\")\n","\n","We can build this circuit in `pytket` by adding gate operations manually:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import Circuit\n","from pytket.circuit.display import render_circuit_jupyter"]},{"cell_type":"markdown","metadata":{},"source":["lets build the QFT for three qubits"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft3_circ = Circuit(3)\n","qft3_circ.H(0)\n","qft3_circ.CU1(0.5, 1, 0)\n","qft3_circ.CU1(0.25, 2, 0)\n","qft3_circ.H(1)\n","qft3_circ.CU1(0.5, 2, 1)\n","qft3_circ.H(2)\n","qft3_circ.SWAP(0, 2)\n","render_circuit_jupyter(qft3_circ)"]},{"cell_type":"markdown","metadata":{},"source":["We can generalise the quantum Fourier transform to $n$ qubits by iterating over the qubits as follows"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def build_qft_circuit(n_qubits: int) -> Circuit:\n"," circ = Circuit(n_qubits, name=\"QFT\")\n"," for i in range(n_qubits):\n"," circ.H(i)\n"," for j in range(i + 1, n_qubits):\n"," circ.CU1(1 / 2 ** (j - i), j, i)\n"," for k in range(0, n_qubits // 2):\n"," circ.SWAP(k, n_qubits - k - 1)\n"," return circ"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft4_circ: Circuit = build_qft_circuit(4)\n","render_circuit_jupyter(qft4_circ)"]},{"cell_type":"markdown","metadata":{},"source":["Now that we have the generalised circuit we can wrap it up in a `CircBox` which can then be added to another circuit as a subroutine."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import CircBox"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft4_box: CircBox = CircBox(qft4_circ)\n","qft_circ = Circuit(4).add_gate(qft4_box, [0, 1, 2, 3])\n","render_circuit_jupyter(qft_circ)"]},{"cell_type":"markdown","metadata":{},"source":["Note how the `CircBox` inherits the name `QFT` from the underlying circuit."]},{"cell_type":"markdown","metadata":{},"source":["Recall that in our phase estimation algorithm we need to use the inverse QFT.\n","\n","$$\n","\\begin{equation}\n","\\text{QFT}^† : \\sum_{k=0}^{N - 1} e^{2 \\pi ijk/N}|k\\rangle \\longmapsto |j\\rangle\\,, \\quad N= 2^k\n","\\end{equation}\n","$$\n","\n","\n","Now that we have the QFT circuit we can obtain the inverse by using `CircBox.dagger`. We can also verify that this is correct by inspecting the circuit inside with `CircBox.get_circuit()`."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["inv_qft4_box = qft4_box.dagger\n","render_circuit_jupyter(inv_qft4_box.get_circuit())"]},{"cell_type":"markdown","metadata":{},"source":["## The Controlled Unitary Operations"]},{"cell_type":"markdown","metadata":{},"source":["In the phase estimation algorithm we repeatedly perform controlled unitary operations. In the canonical variant, the number of controlled unitaries will be $2^m - 1$ where $m$ is the number of measurement qubits."]},{"cell_type":"markdown","metadata":{},"source":["The form of $U$ will vary depending on the application. For chemistry or condensed matter physics $U$ typically be the time evolution operator $U(t) = e^{- i H t}$ where $H$ is the problem Hamiltonian."]},{"cell_type":"markdown","metadata":{},"source":["Suppose that we had the following decomposition for $H$ in terms of Pauli strings $P_j$ and complex coefficients $\\alpha_j$.\n","\n","\\begin{equation}\n","H = \\sum_j \\alpha_j P_j\\,, \\quad \\, P_j \\in \\{I, X, Y, Z\\}^{\\otimes n}\n","\\end{equation}\n","\n","Here Pauli strings refers to tensor products of Pauli operators. These strings form an orthonormal basis for $2^n \\times 2^n$ matrices."]},{"cell_type":"markdown","metadata":{},"source":["If we have a Hamiltonian in the form above, we can then implement $U(t)$ as a sequence of Pauli gadget circuits. We can do this with the [PauliExpBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.PauliExpBox) construct in pytket. For more on `PauliExpBox` see the [user manual](https://tket.quantinuum.com/user-manual/manual_circuit.html#pauli-exponential-boxes)."]},{"cell_type":"markdown","metadata":{},"source":["Once we have a circuit to implement our time evolution operator $U(t)$, we can construct the controlled $U(t)$ operations using [QControlBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.QControlBox). If our base unitary is a sequence of `PauliExpBox`(es) then there is some structure we can exploit to simplify our circuit. See this [blog post](https://tket.quantinuum.com/tket-blog/posts/controlled_gates/) on [ConjugationBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.ConjugationBox) for more."]},{"cell_type":"markdown","metadata":{},"source":["In what follows, we will just construct a simplified instance of QPE where the controlled unitaries are just $\\text{CU1}$ gates."]},{"cell_type":"markdown","metadata":{},"source":["## Putting it all together"]},{"cell_type":"markdown","metadata":{},"source":["We can now define a function to build our entire QPE circuit. We can make this function take a state preparation circuit and a unitary circuit as input as well. The function also has the number of measurement qubits as input which will determine the precision of our phase estimate."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import QControlBox"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def build_phase_est_circuit(\n"," n_measurement_qubits: int, state_prep_circuit: Circuit, unitary_circuit: Circuit\n",") -> Circuit:\n"," qpe_circ: Circuit = Circuit()\n"," n_state_prep_qubits = state_prep_circuit.n_qubits\n"," measurement_register = qpe_circ.add_q_register(\"m\", n_measurement_qubits)\n"," state_prep_register = qpe_circ.add_q_register(\"p\", n_state_prep_qubits)\n"," qpe_circ.add_circuit(state_prep_circuit, list(state_prep_register))\n","\n"," # Create a controlled unitary with a single control qubit\n"," unitary_circuit.name = \"U\"\n"," controlled_u_gate = QControlBox(CircBox(unitary_circuit), 1)\n","\n"," # Add Hadamard gates to every qubit in the measurement register\n"," for m_qubit in measurement_register:\n"," qpe_circ.H(m_qubit)\n","\n"," # Add all (2**n_measurement_qubits - 1) of the controlled unitaries sequentially\n"," for m_qubit in range(n_measurement_qubits):\n"," control_index = n_measurement_qubits - m_qubit - 1\n"," control_qubit = [measurement_register[control_index]]\n"," for _ in range(2**m_qubit):\n"," qpe_circ.add_qcontrolbox(\n"," controlled_u_gate, control_qubit + list(state_prep_register)\n"," )\n","\n"," # Finally, append the inverse qft and measure the qubits\n"," qft_box = CircBox(build_qft_circuit(n_measurement_qubits))\n"," inverse_qft_box = qft_box.dagger\n"," qpe_circ.add_circbox(inverse_qft_box, list(measurement_register))\n"," qpe_circ.measure_register(measurement_register, \"c\")\n"," return qpe_circ"]},{"cell_type":"markdown","metadata":{},"source":["## Phase Estimation with a Trivial Eigenstate\n","\n","Lets test our circuit construction by preparing a trivial $|1\\rangle$ eigenstate of the $\\text{U1}$ gate. We can then see if our phase estimation circuit returns the expected eigenvalue."]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","U1(\\phi)|1\\rangle = e^{i\\phi} = e^{2 \\pi i \\theta} \\implies \\theta = \\frac{\\phi}{2}\n","\\end{equation}\n","$$\n","\n","So we expect that our ideal phase $\\theta$ will be half the input angle $\\phi$ to our $U1$ gate."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["prep_circuit = Circuit(1).X(0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["input_angle = 0.73 # angle as number of half turns"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["unitary_circuit = Circuit(1).U1(input_angle, 0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qpe_circ_trivial = build_phase_est_circuit(\n"," 4, state_prep_circuit=prep_circuit, unitary_circuit=unitary_circuit\n",")"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["render_circuit_jupyter(qpe_circ_trivial)"]},{"cell_type":"markdown","metadata":{},"source":["Lets use the noiseless `AerBackend` simulator to run our phase estimation circuit."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.extensions.qiskit import AerBackend"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["backend = AerBackend()"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["compiled_circ = backend.get_compiled_circuit(qpe_circ_trivial)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_shots = 1000\n","result = backend.run_circuit(compiled_circ, n_shots)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(result.get_counts())"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.backends.backendresult import BackendResult\n","import matplotlib.pyplot as plt"]},{"cell_type":"markdown","metadata":{},"source":["plotting function for QPE Notebook"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def plot_qpe_results(\n"," sim_result: BackendResult,\n"," n_strings: int = 4,\n"," dark_mode: bool = False,\n"," y_limit: int = 1000,\n",") -> None:\n"," \"\"\"\n"," Plots results in a barchart given a BackendResult. the number of stings displayed\n"," can be specified with the n_strings argument.\n"," \"\"\"\n"," counts_dict = sim_result.get_counts()\n"," sorted_shots = counts_dict.most_common()\n"," n_most_common_strings = sorted_shots[:n_strings]\n"," x_axis_values = [str(entry[0]) for entry in n_most_common_strings] # basis states\n"," y_axis_values = [entry[1] for entry in n_most_common_strings] # counts\n"," if dark_mode:\n"," plt.style.use(\"dark_background\")\n"," fig = plt.figure()\n"," ax = fig.add_axes((0, 0, 0.75, 0.5))\n"," color_list = [\"orange\"] * (len(x_axis_values))\n"," ax.bar(\n"," x=x_axis_values,\n"," height=y_axis_values,\n"," color=color_list,\n"," )\n"," ax.set_title(label=\"Results\")\n"," plt.ylim([0, y_limit])\n"," plt.xlabel(\"Basis State\")\n"," plt.ylabel(\"Number of Shots\")\n"," plt.xticks(rotation=90)\n"," plt.show()"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plot_qpe_results(result, y_limit=int(1.2 * n_shots))"]},{"cell_type":"markdown","metadata":{},"source":["As expected we see one outcome with high probability. Lets now extract our approximation of $\\theta$ from our output bitstrings.\n","\n","suppose the $j$ is an integer representation of our most commonly measured bitstring."]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","\\theta_{estimate} = \\frac{j}{N}\n","\\end{equation}\n","$$"]},{"cell_type":"markdown","metadata":{},"source":["Here $N = 2 ^n$ where $n$ is the number of measurement qubits."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.backends.backendresult import BackendResult"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def single_phase_from_backendresult(result: BackendResult) -> float:\n"," # Extract most common measurement outcome\n"," basis_state = result.get_counts().most_common()[0][0]\n"," bitstring = \"\".join([str(bit) for bit in basis_state])\n"," integer = int(bitstring, 2)\n","\n"," # Calculate theta estimate\n"," return integer / (2 ** len(bitstring))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["theta = single_phase_from_backendresult(result)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(theta)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(input_angle / 2)"]},{"cell_type":"markdown","metadata":{},"source":["Our output is close to half our input angle $\\phi$ as expected. Lets calculate our error."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["error = round(abs(input_angle - (2 * theta)), 3)\n","print(error)"]},{"cell_type":"markdown","metadata":{},"source":["## Suggestions for further reading\n","\n","In this notebook we have shown the canonical variant of quantum phase estimation. There are several other variants.\n","\n","Quantinuum paper on Bayesian phase estimation -> https://arxiv.org/pdf/2306.16608.pdf\n","Blog post on `ConjugationBox` -> https://tket.quantinuum.com/tket-blog/posts/controlled_gates/ - efficient circuits for controlled Pauli gadgets.\n","\n","As mentioned quantum phase estimation is a subroutine in Shor's algorithm. Read more about how phase estimation is used in period finding."]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} +{"cells":[{"cell_type":"markdown","metadata":{},"source":["# Quantum Phase Estimation\n","\n","When constructing circuits for quantum algorithms it is useful to think of higher level operations than just individual quantum gates.\n","\n","In `pytket` we can construct circuits using box structures which abstract away the complexity of the underlying circuit.\n","\n","This notebook is intended to complement the [boxes section](https://tket.quantinuum.com/user-manual/manual_circuit.html#boxes) of the user manual which introduces the different box types.\n","\n","To demonstrate boxes in `pytket` we will consider the Quantum Phase Estimation algorithm (QPE). This is an important subroutine in several quantum algorithms including Shor's algorithm and fault-tolerant approaches to quantum chemistry.\n","\n","## Overview of Phase Estimation\n","\n","The Quantum Phase Estimation algorithm can be used to estimate the eigenvalues of some unitary operator $U$ to some desired precision.\n","\n","The eigenvalues of $U$ lie on the unit circle, giving us the following eigenvalue equation\n","\n","$$\n","\\begin{equation}\n","U |\\psi \\rangle = e^{2 \\pi i \\theta} |\\psi\\rangle\\,, \\quad 0 \\leq \\theta \\leq 1\n","\\end{equation}\n","$$\n","\n","Here $|\\psi \\rangle$ is an eigenstate of the operator $U$. In phase estimation we estimate the eigenvalue $e^{2 \\pi i \\theta}$ by approximating $\\theta$.\n","\n","\n","The circuit for Quantum phase estimation is itself composed of several subroutines which we can realise as boxes.\n","\n","![](images/phase_est.png \"Quantum Phase Estimation Circuit\")"]},{"cell_type":"markdown","metadata":{},"source":["QPE is generally split up into three stages\n","\n","1. Firstly we prepare an initial state in one register. In parallel we prepare a uniform superposition state using Hadamard gates on some ancilla qubits. The number of ancilla qubits determines how precisely we can estimate the phase $\\theta$.\n","\n","2. Secondly we apply successive controlled $U$ gates. This has the effect of \"kicking back\" phases onto the ancilla qubits according to the eigenvalue equation above.\n","\n","3. Finally we apply the inverse Quantum Fourier Transform (QFT). This essentially plays the role of destructive interference, suppressing amplitudes from \"undesirable states\" and hopefully allowing us to measure a single outcome (or a small number of outcomes) with high probability.\n","\n","\n","There is some subtlety around the first point. The initial state used can be an exact eigenstate of $U$ however this may be difficult to prepare if we don't know the eigenvalues of $U$ in advance. Alternatively we could use an initial state that is a linear combination of eigenstates, as the phase estimation will project into the eigenspace of $U$."]},{"cell_type":"markdown","metadata":{},"source":["We also assume that we can implement $U$ with a quantum circuit. In chemistry applications $U$ could be of the form $U=e^{-iHt}$ where $H$ is the Hamiltonian of some system of interest. In the cannonical algorithm, the number of controlled unitaries we apply scales exponentially with the number of ancilla qubits. This allows more precision at the expense of a larger quantum circuit."]},{"cell_type":"markdown","metadata":{},"source":["## The Quantum Fourier Transform"]},{"cell_type":"markdown","metadata":{},"source":["Before considering the other parts of the QPE algorithm, lets focus on the Quantum Fourier Transform (QFT) subroutine.\n","\n","Mathematically, the QFT has the following action.\n","\n","$$\n","\\begin{equation}\n","QFT : |j\\rangle\\ \\longmapsto \\sum_{k=0}^{N - 1} e^{2 \\pi ijk/N}|k\\rangle, \\quad N= 2^k\n","\\end{equation}\n","$$\n","\n","This is essentially the Discrete Fourier transform except the input is a quantum state $|j\\rangle$.\n","\n","It is well known that the QFT can be implemented efficiently with a quantum circuit\n","\n","We can build the circuit for the $n$ qubit QFT using $n$ Hadamard gates $\\frac{n}{2}$ swap gates and $\\frac{n(n-1)}{2}$ controlled unitary rotations $\\text{CU1}$.\n","\n","$$\n"," \\begin{equation}\n"," CU1(\\phi) =\n"," \\begin{pmatrix}\n"," I & 0 \\\\\n"," 0 & U1(\\phi)\n"," \\end{pmatrix}\n"," \\,, \\quad\n","U1(\\phi) =\n"," \\begin{pmatrix}\n"," 1 & 0 \\\\\n"," 0 & e^{i \\phi}\n"," \\end{pmatrix}\n"," \\end{equation}\n","$$\n","\n","The circuit for the Quantum Fourier transform on three qubits is the following\n","\n","![](images/qft.png \"QFT Circuit\")\n","\n","We can build this circuit in `pytket` by adding gate operations manually:"]},{"cell_type":"markdown","metadata":{},"source":["lets build the QFT for three qubits"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import Circuit\n","from pytket.circuit.display import render_circuit_jupyter"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft3_circ = Circuit(3)\n","qft3_circ.H(0)\n","qft3_circ.CU1(0.5, 1, 0)\n","qft3_circ.CU1(0.25, 2, 0)\n","qft3_circ.H(1)\n","qft3_circ.CU1(0.5, 2, 1)\n","qft3_circ.H(2)\n","qft3_circ.SWAP(0, 2)\n","render_circuit_jupyter(qft3_circ)"]},{"cell_type":"markdown","metadata":{},"source":["We can generalise the quantum Fourier transform to $n$ qubits by iterating over the qubits as follows"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def build_qft_circuit(n_qubits: int) -> Circuit:\n"," circ = Circuit(n_qubits, name=\"QFT\")\n"," for i in range(n_qubits):\n"," circ.H(i)\n"," for j in range(i + 1, n_qubits):\n"," circ.CU1(1 / 2 ** (j - i), j, i)\n"," for k in range(0, n_qubits // 2):\n"," circ.SWAP(k, n_qubits - k - 1)\n"," return circ"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft4_circ: Circuit = build_qft_circuit(4)\n","render_circuit_jupyter(qft4_circ)"]},{"cell_type":"markdown","metadata":{},"source":["Now that we have the generalised circuit we can wrap it up in a `CircBox` which can then be added to another circuit as a subroutine."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import CircBox"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft4_box: CircBox = CircBox(qft4_circ)\n","qft_circ = Circuit(4).add_gate(qft4_box, [0, 1, 2, 3])\n","render_circuit_jupyter(qft_circ)"]},{"cell_type":"markdown","metadata":{},"source":["Note how the `CircBox` inherits the name `QFT` from the underlying circuit."]},{"cell_type":"markdown","metadata":{},"source":["Recall that in our phase estimation algorithm we need to use the inverse QFT.\n","\n","$$\n","\\begin{equation}\n","\\text{QFT}^† : \\sum_{k=0}^{N - 1} e^{2 \\pi ijk/N}|k\\rangle \\longmapsto |j\\rangle\\,, \\quad N= 2^k\n","\\end{equation}\n","$$\n","\n","\n","Now that we have the QFT circuit we can obtain the inverse by using `CircBox.dagger`. We can also verify that this is correct by inspecting the circuit inside with `CircBox.get_circuit()`."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["inv_qft4_box = qft4_box.dagger\n","render_circuit_jupyter(inv_qft4_box.get_circuit())"]},{"cell_type":"markdown","metadata":{},"source":["## The Controlled Unitary Operations"]},{"cell_type":"markdown","metadata":{},"source":["In the phase estimation algorithm we repeatedly perform controlled unitary operations. In the canonical variant, the number of controlled unitaries will be $2^m - 1$ where $m$ is the number of measurement qubits."]},{"cell_type":"markdown","metadata":{},"source":["The form of $U$ will vary depending on the application. For chemistry or condensed matter physics $U$ typically be the time evolution operator $U(t) = e^{- i H t}$ where $H$ is the problem Hamiltonian."]},{"cell_type":"markdown","metadata":{},"source":["Suppose that we had the following decomposition for $H$ in terms of Pauli strings $P_j$ and complex coefficients $\\alpha_j$.\n","\n","$$\n","\\begin{equation}\n","H = \\sum_j \\alpha_j P_j\\,, \\quad \\, P_j \\in \\{I, \\,X, \\,Y, \\,Z\\}^{\\otimes n}\n","\\end{equation}\n","$$\n","\n","Here Pauli strings refers to tensor products of Pauli operators. These strings form an orthonormal basis for $2^n \\times 2^n$ matrices."]},{"cell_type":"markdown","metadata":{},"source":["If we have a Hamiltonian in the form above, we can then implement $U(t)$ as a sequence of Pauli gadget circuits. We can do this with the [PauliExpBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.PauliExpBox) construct in pytket. For more on `PauliExpBox` see the [user manual](https://tket.quantinuum.com/user-manual/manual_circuit.html#pauli-exponential-boxes)."]},{"cell_type":"markdown","metadata":{},"source":["Once we have a circuit to implement our time evolution operator $U(t)$, we can construct the controlled $U(t)$ operations using [QControlBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.QControlBox). If our base unitary is a sequence of `PauliExpBox`(es) then there is some structure we can exploit to simplify our circuit. See this [blog post](https://tket.quantinuum.com/tket-blog/posts/controlled_gates/) on [ConjugationBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.ConjugationBox) for more."]},{"cell_type":"markdown","metadata":{},"source":["In what follows, we will just construct a simplified instance of QPE where the controlled unitaries are just $\\text{CU1}$ gates."]},{"cell_type":"markdown","metadata":{},"source":["## Putting it all together"]},{"cell_type":"markdown","metadata":{},"source":["We can now define a function to build our entire QPE circuit. We can make this function take a state preparation circuit and a unitary circuit as input as well. The function also has the number of measurement qubits as input which will determine the precision of our phase estimate."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import QControlBox"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def build_phase_est_circuit(\n"," n_measurement_qubits: int, state_prep_circuit: Circuit, unitary_circuit: Circuit\n",") -> Circuit:\n"," qpe_circ: Circuit = Circuit()\n"," n_state_prep_qubits = state_prep_circuit.n_qubits\n"," measurement_register = qpe_circ.add_q_register(\"m\", n_measurement_qubits)\n"," state_prep_register = qpe_circ.add_q_register(\"p\", n_state_prep_qubits)\n"," qpe_circ.add_circuit(state_prep_circuit, list(state_prep_register))\n","\n"," # Create a controlled unitary with a single control qubit\n"," unitary_circuit.name = \"U\"\n"," controlled_u_gate = QControlBox(CircBox(unitary_circuit), 1)\n","\n"," # Add Hadamard gates to every qubit in the measurement register\n"," for m_qubit in measurement_register:\n"," qpe_circ.H(m_qubit)\n","\n"," # Add all (2**n_measurement_qubits - 1) of the controlled unitaries sequentially\n"," for m_qubit in range(n_measurement_qubits):\n"," control_index = n_measurement_qubits - m_qubit - 1\n"," control_qubit = [measurement_register[control_index]]\n"," for _ in range(2**m_qubit):\n"," qpe_circ.add_qcontrolbox(\n"," controlled_u_gate, control_qubit + list(state_prep_register)\n"," )\n","\n"," # Finally, append the inverse qft and measure the qubits\n"," qft_box = CircBox(build_qft_circuit(n_measurement_qubits))\n"," inverse_qft_box = qft_box.dagger\n"," qpe_circ.add_circbox(inverse_qft_box, list(measurement_register))\n"," qpe_circ.measure_register(measurement_register, \"c\")\n"," return qpe_circ"]},{"cell_type":"markdown","metadata":{},"source":["## Phase Estimation with a Trivial Eigenstate\n","\n","Lets test our circuit construction by preparing a trivial $|1\\rangle$ eigenstate of the $\\text{U1}$ gate. We can then see if our phase estimation circuit returns the expected eigenvalue."]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","U1(\\phi)|1\\rangle = e^{i\\phi} = e^{2 \\pi i \\theta} \\implies \\theta = \\frac{\\phi}{2}\n","\\end{equation}\n","$$\n","\n","So we expect that our ideal phase $\\theta$ will be half the input angle $\\phi$ to our $U1$ gate."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["prep_circuit = Circuit(1).X(0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["input_angle = 0.73 # angle as number of half turns"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["unitary_circuit = Circuit(1).U1(input_angle, 0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qpe_circ_trivial = build_phase_est_circuit(\n"," 4, state_prep_circuit=prep_circuit, unitary_circuit=unitary_circuit\n",")"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["render_circuit_jupyter(qpe_circ_trivial)"]},{"cell_type":"markdown","metadata":{},"source":["Lets use the noiseless `AerBackend` simulator to run our phase estimation circuit."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.extensions.qiskit import AerBackend"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["backend = AerBackend()"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["compiled_circ = backend.get_compiled_circuit(qpe_circ_trivial)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_shots = 1000\n","result = backend.run_circuit(compiled_circ, n_shots)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(result.get_counts())"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.backends.backendresult import BackendResult\n","import matplotlib.pyplot as plt"]},{"cell_type":"markdown","metadata":{},"source":["plotting function for QPE Notebook"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def plot_qpe_results(\n"," sim_result: BackendResult,\n"," n_strings: int = 4,\n"," dark_mode: bool = False,\n"," y_limit: int = 1000,\n",") -> None:\n"," \"\"\"\n"," Plots results in a barchart given a BackendResult. the number of stings displayed\n"," can be specified with the n_strings argument.\n"," \"\"\"\n"," counts_dict = sim_result.get_counts()\n"," sorted_shots = counts_dict.most_common()\n"," n_most_common_strings = sorted_shots[:n_strings]\n"," x_axis_values = [str(entry[0]) for entry in n_most_common_strings] # basis states\n"," y_axis_values = [entry[1] for entry in n_most_common_strings] # counts\n"," if dark_mode:\n"," plt.style.use(\"dark_background\")\n"," fig = plt.figure()\n"," ax = fig.add_axes((0, 0, 0.75, 0.5))\n"," color_list = [\"orange\"] * (len(x_axis_values))\n"," ax.bar(\n"," x=x_axis_values,\n"," height=y_axis_values,\n"," color=color_list,\n"," )\n"," ax.set_title(label=\"Results\")\n"," plt.ylim([0, y_limit])\n"," plt.xlabel(\"Basis State\")\n"," plt.ylabel(\"Number of Shots\")\n"," plt.show()"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plot_qpe_results(result, y_limit=int(1.2 * n_shots))"]},{"cell_type":"markdown","metadata":{},"source":["As expected we see one outcome with high probability. Lets now extract our approximation of $\\theta$ from our output bitstrings.\n","\n","suppose the $j$ is an integer representation of our most commonly measured bitstring."]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","\\theta_{estimate} = \\frac{j}{N}\n","\\end{equation}\n","$$"]},{"cell_type":"markdown","metadata":{},"source":["Here $N = 2 ^n$ where $n$ is the number of measurement qubits."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.backends.backendresult import BackendResult"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def single_phase_from_backendresult(result: BackendResult) -> float:\n"," # Extract most common measurement outcome\n"," basis_state = result.get_counts().most_common()[0][0]\n"," bitstring = \"\".join([str(bit) for bit in basis_state])\n"," integer = int(bitstring, 2)\n","\n"," # Calculate theta estimate\n"," return integer / (2 ** len(bitstring))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["theta = single_phase_from_backendresult(result)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(theta)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(input_angle / 2)"]},{"cell_type":"markdown","metadata":{},"source":["Our output is close to half our input angle $\\phi$ as expected. Lets calculate our error."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["error = round(abs(input_angle - (2 * theta)), 3)\n","print(error)"]},{"cell_type":"markdown","metadata":{},"source":["## Suggestions for further reading\n","\n","In this notebook we have shown the canonical variant of quantum phase estimation. There are several other variants.\n","\n","Quantinuum paper on Bayesian phase estimation -> https://arxiv.org/pdf/2306.16608.pdf\n","Blog post on `ConjugationBox` -> https://tket.quantinuum.com/tket-blog/posts/controlled_gates/ - efficient circuits for controlled Pauli gadgets.\n","\n","As mentioned quantum phase estimation is a subroutine in Shor's algorithm. Read more about how phase estimation is used in period finding."]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} diff --git a/examples/python/phase_estimation.py b/examples/python/phase_estimation.py index 1eaf3937..dcafd83c 100644 --- a/examples/python/phase_estimation.py +++ b/examples/python/phase_estimation.py @@ -1,7 +1,4 @@ -#!/usr/bin/env python -# coding: utf-8 - -# # Quantum Phase Estimation using `pytket` Boxes +# # Quantum Phase Estimation # # When constructing circuits for quantum algorithms it is useful to think of higher level operations than just individual quantum gates. # @@ -49,9 +46,11 @@ # # Mathematically, the QFT has the following action. # +# $$ # \begin{equation} # QFT : |j\rangle\ \longmapsto \sum_{k=0}^{N - 1} e^{2 \pi ijk/N}|k\rangle, \quad N= 2^k # \end{equation} +# $$ # # This is essentially the Discrete Fourier transform except the input is a quantum state $|j\rangle$. # @@ -81,11 +80,10 @@ # # We can build this circuit in `pytket` by adding gate operations manually: - +# lets build the QFT for three qubits from pytket.circuit import Circuit from pytket.circuit.display import render_circuit_jupyter -# lets build the QFT for three qubits qft3_circ = Circuit(3) qft3_circ.H(0) qft3_circ.CU1(0.5, 1, 0) @@ -152,9 +150,11 @@ def build_qft_circuit(n_qubits: int) -> Circuit: # Suppose that we had the following decomposition for $H$ in terms of Pauli strings $P_j$ and complex coefficients $\alpha_j$. # +# $$ # \begin{equation} -# H = \sum_j \alpha_j P_j\,, \quad \, P_j \in \{I, X, Y, Z\}^{\otimes n} +# H = \sum_j \alpha_j P_j\,, \quad \, P_j \in \{I, \,X, \,Y, \,Z\}^{\otimes n} # \end{equation} +# $$ # # Here Pauli strings refers to tensor products of Pauli operators. These strings form an orthonormal basis for $2^n \times 2^n$ matrices. @@ -291,7 +291,6 @@ def plot_qpe_results( plt.ylim([0, y_limit]) plt.xlabel("Basis State") plt.ylabel("Number of Shots") - plt.xticks(rotation=90) plt.show() From cb6f945d8eb708410dedf43a6f794ed740ba6874 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Tue, 7 Nov 2023 11:22:07 +0000 Subject: [PATCH 35/51] delete legacy benchamarking example --- examples/_toc.yml | 2 - examples/benchmarking/README.md | 6 - examples/images/IBMTokyoArc.png | Bin 63985 -> 0 bytes examples/images/IBMqx5Arc.png | Bin 36325 -> 0 bytes examples/tket_benchmarking.ipynb | 852 ------------------------------- 5 files changed, 860 deletions(-) delete mode 100644 examples/benchmarking/README.md delete mode 100644 examples/images/IBMTokyoArc.png delete mode 100644 examples/images/IBMqx5Arc.png delete mode 100644 examples/tket_benchmarking.ipynb diff --git a/examples/_toc.yml b/examples/_toc.yml index a4311891..c2227cb8 100644 --- a/examples/_toc.yml +++ b/examples/_toc.yml @@ -33,8 +33,6 @@ parts: - file: pytket-qujax_heisenberg_vqe - caption: Other chapters: - - file: benchmarking/README - - file: tket_benchmarking - file: entanglement_swapping - file: spam_example - file: expectation_value_example diff --git a/examples/benchmarking/README.md b/examples/benchmarking/README.md deleted file mode 100644 index ba3f3ec8..00000000 --- a/examples/benchmarking/README.md +++ /dev/null @@ -1,6 +0,0 @@ -# Benchmarking - -The circuits used for the benchmarking experiments reported in -[arxiv:1902:08091](https://arxiv.org/abs/1902.08091) are available in the -[tket_benchmarking](https://github.com/CQCL/tket_benchmarking) repository -(in the folder `arxiv-1902-08091`). diff --git a/examples/images/IBMTokyoArc.png b/examples/images/IBMTokyoArc.png deleted file mode 100644 index 1204042237fb64cf2de98085dbf4c9e13ab9c82e..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 63985 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 B`UrvvjTjQnn+ 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 diff --git a/examples/images/IBMqx5Arc.png b/examples/images/IBMqx5Arc.png deleted file mode 100644 index 3ca7b1f46776ba72bda1ff87491b7a5c50fef53c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 36325 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 diff --git a/examples/tket_benchmarking.ipynb b/examples/tket_benchmarking.ipynb deleted file mode 100644 index 29411e83..00000000 --- a/examples/tket_benchmarking.ipynb +++ /dev/null @@ -1,852 +0,0 @@ -{ - "cells": [ - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "# TKET benchmarking example\n" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "Warning: This notebook is now deprecated for versions from 0.2.0 onwards, and is not going to be updated in ongoing releases.\n", - "\n", - "The aim of this example is to show how to run the IBM benchmarking circuits through tket. You will need both `pytket` and `pytket_qiskit` installed from pip before running this turoial. You will also need `pandas` to capture the data.\n", - "\n", - "The benchmarking circuits originated from https://github.com/iic-jku/ibm_qx_mapping/tree/master/examples, but there is a copy in pytket in the \"benchmarking\" folder. The initial circuits are written in QASM, meaning that they must be converted to tket's internal representation via Qiskit. Using this script we will compile these circuits through tket and then print a table to terminal containing analysis of the circuits post-compilation." - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "We need access to Qiskit's `QuantumCircuit` and the `circuit_to_dag` converter, which will allow us to take the original QASM and retrieve a Qiskit DAGCircuit. There is then a pytket method `dagcircuit_to_tk` to convert this DAGCircuit to tket's internal representation. We also need the pytket method `coupling_to_arc` to convert IBM's architectures to the tket `Architecture` class for the use of routing." - ] - }, - { - "cell_type": "code", - "execution_count": 2, - "metadata": {}, - "outputs": [], - "source": [ - "from qiskit import QuantumCircuit\n", - "\n", - "from pytket.extensions.qiskit import qiskit_to_tk" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "We need all of the required pytket equipment to allow us to perform clean-up transform passes, to route and to analyse the circuits. Lastly, we require the `pandas` module to hold our data, and \n", - "`time` to benchmark compilation time per circuit." - ] - }, - { - "cell_type": "code", - "execution_count": 3, - "metadata": {}, - "outputs": [], - "source": [ - "from pytket import Architecture, route\n", - "from pytket import OpType, Transform\n", - "import pandas\n", - "import time" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "The architectures used to benchmark for our routing paper were the IBMqx5 and IBMTokyo architectures. The architectures qx5 and Tokyo respectively are shown below (although the diagrams do not show the directedness of the coupling maps). These diagrams are from https://www.research.ibm.com/ibm-q/technology/devices/#ibmqx5. We will now define the coupling map representing both architectures. These will later be converted to directed graphs to be used by tket.\n", - "![alt text](IBMqx5Arc.png \"Title\")\n", - "![alt text](IBMTokyoArc.png \"Title2\")" - ] - }, - { - "cell_type": "code", - "execution_count": 4, - "metadata": {}, - "outputs": [], - "source": [ - "ibm_devices = {\n", - " \"ibmqx5\": {\"edges\": [(1, 0), (1, 2), (2, 3), (3, 4), \n", - " (3, 14), (5, 4), (6, 5), (6, 7), (6, 11), (7, 10), \n", - " (8, 7), (9, 8), (9, 10), (11, 10), (12, 5), (12, 11), \n", - " (12, 13), (13, 4), (13, 14), (15, 0), (15, 2), (15, 14)],\n", - " \"nodes\": 16},\n", - " \"ibmq_20_tokyo\": {\"edges\": [(0, 1), (0, 5), (1, 2), (1, 6), (1, 7), (2, 3), (2, 6),\n", - " (2, 7), (3, 4), (3, 8), (3, 9), (4, 8), (4, 9), (5, 6), (5, 10), (5, 11), (6, 7),\n", - " (6, 10), (6, 11), (7, 8), (7, 12), (7, 13), (8, 9), (8, 12), (8, 13), (9, 14), (10, 11),\n", - " (10, 15), (11, 12), (11, 16), (11, 17), (12, 13), (12, 16), (12, 17), (13, 14), (13, 18),\n", - " (13, 19), (14, 18), (14, 19), (15, 16), (16, 17), (17, 18), (18, 19)], \"nodes\": 20}\n", - "}" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "We shall now choose the device and create a directed graph for tket's routing to use." - ] - }, - { - "cell_type": "code", - "execution_count": 5, - "metadata": {}, - "outputs": [], - "source": [ - "device_name = 'ibmqx5' #'ibm_20_tokyo' ###Note: can also be ran using the Tokyo machine architecture\n", - " ### or with a user-defined coupling map\n", - "coupling_map = ibm_devices[device_name][\"edges\"]\n", - "directed_arc = Architecture(coupling_map)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "We now define a method which takes in (1) a QASM file and (2) a directed graph architecture; it returns analysis of the circuit after our transform passes and routing procedure have been completed. It will also print out to terminal the time taken for all the transformations and routing to finish. Changing which optimisations are run in this method will trade off quality of the ouputs for time taken. For example, removing all optimisation passes and just running the routing procedure will give the fastest run time but may leave some redundant gates in the final circuit." - ] - }, - { - "cell_type": "code", - "execution_count": 6, - "metadata": {}, - "outputs": [], - "source": [ - "def getStats(filename, directed_arc):\n", - " qc = QuantumCircuit.from_qasm_file(filename)\n", - " tkcirc = qiskit_to_tk(qc)\n", - " start_time = time.process_time()\n", - " Transform.OptimisePhaseGadgets().apply(tkcirc)\n", - " outcirc = route(tkcirc, directed_arc)\n", - " # decompose swaps to CX gates and redirect CXs in wrong direction\n", - " outcirc.decompose_SWAP_to_CX()\n", - " outcirc.redirect_CX_gates(directed_arc)\n", - " Transform.OptimisePostRouting().apply(outcirc)\n", - " \n", - " time_elapsed = time.process_time() - start_time\n", - " \n", - " print(\"Compilation time for circuit \" + str(filename) + \": \" + str(time_elapsed) + \"s\")\n", - " if outcirc.n_gates==0:\n", - " return [0,0,0,0,0]\n", - " ###Returns: [number of vertices, circuit depth, nubmer of CX gates, number of parallel slices of CX gates]\n", - " return [outcirc.n_gates, outcirc.depth(), outcirc.n_gates_of_type(OpType.CX), \n", - " outcirc.depth_by_type(OpType.CX), time_elapsed]\n", - " ###Note: the raw number of vertices in the circuits and the raw depth \n", - " ### need to have the i/o vertices removed for fair comparisons" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "Lastly, we generate the table of input QASM filenames from a csv file using `pandas` and run the circuits through our compiler. The results are printed to terminal by default." - ] - }, - { - "cell_type": "code", - "execution_count": 7, - "metadata": {}, - "outputs": [ - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Compilation time for circuit benchmarking/ibmq/xor5_254.qasm: 0.00951000000000013s\n", - "Time spent compiling so far: 0.00951000000000013\n", - "Compilation time for circuit benchmarking/ibmq/graycode6_47.qasm: 0.005398000000000014s\n", - "Time spent compiling so far: 0.014908000000000143\n", - "Compilation time for circuit benchmarking/ibmq/ex1_226.qasm: 0.008242000000000083s\n", - "Time spent compiling so far: 0.023150000000000226\n", - "Compilation time for circuit benchmarking/ibmq/4gt11_84.qasm: 0.015388999999999875s\n", - "Time spent compiling so far: 0.0385390000000001\n", - "Compilation time for circuit benchmarking/ibmq/4mod5-v0_20.qasm: 0.020800999999999625s\n", - "Time spent compiling so far: 0.059339999999999726\n", - "Compilation time for circuit benchmarking/ibmq/ex-1_166.qasm: 0.017431999999999892s\n", - "Time spent compiling so far: 0.07677199999999962\n", - "Compilation time for circuit benchmarking/ibmq/4mod5-v1_22.qasm: 0.01641899999999996s\n", - "Time spent compiling so far: 0.09319099999999958\n", - "Compilation time for circuit benchmarking/ibmq/mod5d1_63.qasm: 0.024310000000000276s\n", - "Time spent compiling so far: 0.11750099999999986\n", - "Compilation time for circuit benchmarking/ibmq/ham3_102.qasm: 0.016796000000000255s\n", - "Time spent compiling so far: 0.1342970000000001\n", - "Compilation time for circuit benchmarking/ibmq/4gt11_83.qasm: 0.01989099999999988s\n", - "Time spent compiling so far: 0.154188\n", - "Compilation time for circuit benchmarking/ibmq/4gt11_82.qasm: 0.024681999999999427s\n", - "Time spent compiling so far: 0.17886999999999942\n", - "Compilation time for circuit benchmarking/ibmq/rd32-v0_66.qasm: 0.022649000000000363s\n", - "Time spent compiling so far: 0.20151899999999978\n", - "Compilation time for circuit benchmarking/ibmq/alu-v0_27.qasm: 0.031051000000000162s\n", - "Time spent compiling so far: 0.23256999999999994\n", - "Compilation time for circuit benchmarking/ibmq/4mod5-v1_24.qasm: 0.035730999999999735s\n", - "Time spent compiling so far: 0.2683009999999997\n", - "Compilation time for circuit benchmarking/ibmq/4mod5-v0_19.qasm: 0.03168000000000015s\n", - "Time spent compiling so far: 0.29998099999999983\n", - "Compilation time for circuit benchmarking/ibmq/mod5mils_65.qasm: 0.033256000000000174s\n", - "Time spent compiling so far: 0.333237\n", - "Compilation time for circuit benchmarking/ibmq/rd32-v1_68.qasm: 0.03151800000000016s\n", - "Time spent compiling so far: 0.36475500000000016\n", - "Compilation time for circuit benchmarking/ibmq/alu-v1_28.qasm: 0.04489699999999974s\n", - "Time spent compiling so far: 0.4096519999999999\n", - "Compilation time for circuit benchmarking/ibmq/alu-v2_33.qasm: 0.03587100000000021s\n", - "Time spent compiling so far: 0.4455230000000001\n", - "Compilation time for circuit benchmarking/ibmq/alu-v4_37.qasm: 0.032544999999999824s\n", - "Time spent compiling so far: 0.47806799999999994\n", - "Compilation time for circuit benchmarking/ibmq/alu-v3_35.qasm: 0.0310940000000004s\n", - "Time spent compiling so far: 0.5091620000000003\n", - "Compilation time for circuit benchmarking/ibmq/3_17_13.qasm: 0.03810599999999997s\n", - "Time spent compiling so far: 0.5472680000000003\n", - "Compilation time for circuit benchmarking/ibmq/alu-v1_29.qasm: 0.033971000000000195s\n", - "Time spent compiling so far: 0.5812390000000005\n", - "Compilation time for circuit benchmarking/ibmq/miller_11.qasm: 0.05386399999999991s\n", - "Time spent compiling so far: 0.6351030000000004\n", - "Compilation time for circuit benchmarking/ibmq/alu-v3_34.qasm: 0.06335800000000003s\n", - "Time spent compiling so far: 0.6984610000000004\n", - "Compilation time for circuit benchmarking/ibmq/decod24-v2_43.qasm: 0.05558200000000024s\n", - "Time spent compiling so far: 0.7540430000000007\n", - "Compilation time for circuit benchmarking/ibmq/decod24-v0_38.qasm: 0.053290000000000504s\n", - "Time spent compiling so far: 0.8073330000000012\n", - "Compilation time for circuit benchmarking/ibmq/mod5d2_64.qasm: 0.04377400000000087s\n", - "Time spent compiling so far: 0.8511070000000021\n", - "Compilation time for circuit benchmarking/ibmq/4gt13_92.qasm: 0.07327100000000009s\n", - "Time spent compiling so far: 0.9243780000000021\n", - "Compilation time for circuit benchmarking/ibmq/4gt13-v1_93.qasm: 0.0683429999999996s\n", - "Time spent compiling so far: 0.9927210000000017\n", - "Compilation time for circuit benchmarking/ibmq/4mod5-v0_18.qasm: 0.05403900000000039s\n", - "Time spent compiling so far: 1.0467600000000021\n", - "Compilation time for circuit benchmarking/ibmq/decod24-bdd_294.qasm: 0.07072400000000023s\n", - "Time spent compiling so far: 1.1174840000000024\n", - "Compilation time for circuit benchmarking/ibmq/one-two-three-v2_100.qasm: 0.0743349999999996s\n", - "Time spent compiling so far: 1.191819000000002\n", - "Compilation time for circuit benchmarking/ibmq/one-two-three-v3_101.qasm: 0.06547600000000031s\n", - "Time spent compiling so far: 1.2572950000000023\n", - "Compilation time for circuit benchmarking/ibmq/4mod5-v1_23.qasm: 0.05346799999999963s\n", - "Time spent compiling so far: 1.310763000000002\n", - "Compilation time for circuit benchmarking/ibmq/4mod5-bdd_287.qasm: 0.07852100000000029s\n", - "Time spent compiling so far: 1.3892840000000022\n", - "Compilation time for circuit benchmarking/ibmq/rd32_270.qasm: 0.061949999999999505s\n", - "Time spent compiling so far: 1.4512340000000017\n", - "Compilation time for circuit benchmarking/ibmq/4gt5_75.qasm: 0.0817629999999987s\n", - "Time spent compiling so far: 1.5329970000000004\n", - "Compilation time for circuit benchmarking/ibmq/alu-bdd_288.qasm: 0.06410000000000071s\n", - "Time spent compiling so far: 1.597097000000001\n", - "Compilation time for circuit benchmarking/ibmq/alu-v0_26.qasm: 0.08118200000000009s\n", - "Time spent compiling so far: 1.6782790000000012\n", - "Compilation time for circuit benchmarking/ibmq/decod24-v1_41.qasm: 0.0812450000000009s\n", - "Time spent compiling so far: 1.759524000000002\n", - "Compilation time for circuit benchmarking/ibmq/rd53_138.qasm: 0.13351399999999902s\n", - "Time spent compiling so far: 1.893038000000001\n", - "Compilation time for circuit benchmarking/ibmq/4gt5_76.qasm: 0.06876499999999997s\n", - "Time spent compiling so far: 1.961803000000001\n", - "Compilation time for circuit benchmarking/ibmq/4gt13_91.qasm: 0.08259399999999939s\n", - "Time spent compiling so far: 2.0443970000000005\n", - "Compilation time for circuit benchmarking/ibmq/cnt3-5_179.qasm: 0.28147000000000055s\n", - "Time spent compiling so far: 2.325867000000001\n", - "Compilation time for circuit benchmarking/ibmq/qft_10.qasm: 0.025814000000000448s\n", - "Time spent compiling so far: 2.325867000000001\n", - "Compilation time for circuit benchmarking/ibmq/4gt13_90.qasm: 0.09255399999999803s\n", - "Time spent compiling so far: 2.418420999999999\n", - "Compilation time for circuit benchmarking/ibmq/alu-v4_36.qasm: 0.11703999999999937s\n", - "Time spent compiling so far: 2.5354609999999984\n", - "Compilation time for circuit benchmarking/ibmq/mini_alu_305.qasm: 0.21232399999999885s\n", - "Time spent compiling so far: 2.7477849999999973\n", - "Compilation time for circuit benchmarking/ibmq/ising_model_10.qasm: 0.2612620000000003s\n", - "Time spent compiling so far: 3.0090469999999976\n", - "Compilation time for circuit benchmarking/ibmq/ising_model_16.qasm: 0.3769770000000001s\n", - "Time spent compiling so far: 3.3860239999999977\n", - "Compilation time for circuit benchmarking/ibmq/ising_model_13.qasm: 0.33491099999999996s\n", - "Time spent compiling so far: 3.7209349999999977\n", - "Compilation time for circuit benchmarking/ibmq/4gt5_77.qasm: 0.1293350000000011s\n", - "Time spent compiling so far: 3.8502699999999987\n", - "Compilation time for circuit benchmarking/ibmq/sys6-v0_111.qasm: 0.30446699999999893s\n", - "Time spent compiling so far: 4.154736999999997\n", - "Compilation time for circuit benchmarking/ibmq/one-two-three-v1_99.qasm: 0.15336699999999936s\n", - "Time spent compiling so far: 4.308103999999997\n", - "Compilation time for circuit benchmarking/ibmq/one-two-three-v0_98.qasm: 0.12628400000000006s\n", - "Time spent compiling so far: 4.434387999999997\n", - "Compilation time for circuit benchmarking/ibmq/decod24-v3_45.qasm: 0.14908700000000152s\n", - "Time spent compiling so far: 4.583474999999998\n", - "Compilation time for circuit benchmarking/ibmq/4gt10-v1_81.qasm: 0.14076600000000106s\n", - "Time spent compiling so far: 4.724240999999999\n", - "Compilation time for circuit benchmarking/ibmq/aj-e11_165.qasm: 0.14970000000000105s\n", - "Time spent compiling so far: 4.873941\n", - "Compilation time for circuit benchmarking/ibmq/4mod7-v0_94.qasm: 0.16386000000000145s\n", - "Time spent compiling so far: 5.037801000000002\n", - "Compilation time for circuit benchmarking/ibmq/alu-v2_32.qasm: 0.15958599999999912s\n", - "Time spent compiling so far: 5.197387000000001\n", - "Compilation time for circuit benchmarking/ibmq/rd73_140.qasm: 0.2630720000000011s\n", - "Time spent compiling so far: 5.460459000000002\n" - ] - }, - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Compilation time for circuit benchmarking/ibmq/4mod7-v1_96.qasm: 0.1598600000000001s\n", - "Time spent compiling so far: 5.620319000000002\n", - "Compilation time for circuit benchmarking/ibmq/4gt4-v0_80.qasm: 0.18252500000000182s\n", - "Time spent compiling so far: 5.802844000000004\n", - "Compilation time for circuit benchmarking/ibmq/mod10_176.qasm: 0.17993299999999834s\n", - "Time spent compiling so far: 5.982777000000002\n", - "Compilation time for circuit benchmarking/ibmq/0410184_169.qasm: 0.6244690000000013s\n", - "Time spent compiling so far: 6.6072460000000035\n", - "Compilation time for circuit benchmarking/ibmq/qft_16.qasm: 0.05671399999999949s\n", - "Time spent compiling so far: 6.6072460000000035\n", - "Compilation time for circuit benchmarking/ibmq/4gt12-v0_88.qasm: 0.20898000000000394s\n", - "Time spent compiling so far: 6.8162260000000074\n", - "Compilation time for circuit benchmarking/ibmq/rd84_142.qasm: 0.7401959999999974s\n", - "Time spent compiling so far: 7.556422000000005\n", - "Compilation time for circuit benchmarking/ibmq/rd53_311.qasm: 0.5026390000000021s\n", - "Time spent compiling so far: 8.059061000000007\n", - "Compilation time for circuit benchmarking/ibmq/4_49_16.qasm: 0.22887300000000366s\n", - "Time spent compiling so far: 8.28793400000001\n", - "Compilation time for circuit benchmarking/ibmq/sym9_146.qasm: 0.5669679999999993s\n", - "Time spent compiling so far: 8.85490200000001\n", - "Compilation time for circuit benchmarking/ibmq/4gt12-v1_89.qasm: 0.24625699999999995s\n", - "Time spent compiling so far: 9.10115900000001\n", - "Compilation time for circuit benchmarking/ibmq/4gt12-v0_87.qasm: 0.21894999999999598s\n", - "Time spent compiling so far: 9.320109000000006\n", - "Compilation time for circuit benchmarking/ibmq/4gt4-v0_79.qasm: 0.19500299999999626s\n", - "Time spent compiling so far: 9.515112000000002\n", - "Compilation time for circuit benchmarking/ibmq/hwb4_49.qasm: 0.23632500000000078s\n", - "Time spent compiling so far: 9.751437000000003\n", - "Compilation time for circuit benchmarking/ibmq/sym6_316.qasm: 0.6572160000000018s\n", - "Time spent compiling so far: 10.408653000000005\n", - "Compilation time for circuit benchmarking/ibmq/4gt12-v0_86.qasm: 0.24052800000000119s\n", - "Time spent compiling so far: 10.649181000000006\n", - "Compilation time for circuit benchmarking/ibmq/4gt4-v0_72.qasm: 0.28336000000000183s\n", - "Time spent compiling so far: 10.932541000000008\n", - "Compilation time for circuit benchmarking/ibmq/4gt4-v0_78.qasm: 0.2004679999999972s\n", - "Time spent compiling so far: 11.133009000000005\n", - "Compilation time for circuit benchmarking/ibmq/mod10_171.qasm: 0.23787799999999848s\n", - "Time spent compiling so far: 11.370887000000003\n", - "Compilation time for circuit benchmarking/ibmq/4gt4-v1_74.qasm: 0.28927499999999995s\n", - "Time spent compiling so far: 11.660162000000003\n", - "Compilation time for circuit benchmarking/ibmq/rd53_135.qasm: 0.37902099999999805s\n", - "Time spent compiling so far: 12.039183000000001\n", - "Compilation time for circuit benchmarking/ibmq/mini-alu_167.qasm: 0.28952900000000525s\n", - "Time spent compiling so far: 12.328712000000007\n", - "Compilation time for circuit benchmarking/ibmq/one-two-three-v0_97.qasm: 0.32423s\n", - "Time spent compiling so far: 12.652942000000007\n", - "Compilation time for circuit benchmarking/ibmq/ham7_104.qasm: 0.37469400000000164s\n", - "Time spent compiling so far: 13.027636000000008\n", - "Compilation time for circuit benchmarking/ibmq/decod24-enable_126.qasm: 0.37364900000000034s\n", - "Time spent compiling so far: 13.401285000000009\n", - "Compilation time for circuit benchmarking/ibmq/mod8-10_178.qasm: 0.43175500000000255s\n", - "Time spent compiling so far: 13.833040000000011\n", - "Compilation time for circuit benchmarking/ibmq/cnt3-5_180.qasm: 2.2587930000000043s\n", - "Time spent compiling so far: 16.091833000000015\n", - "Compilation time for circuit benchmarking/ibmq/ex3_229.qasm: 0.4237730000000006s\n", - "Time spent compiling so far: 16.515606000000016\n", - "Compilation time for circuit benchmarking/ibmq/4gt4-v0_73.qasm: 0.4361159999999984s\n", - "Time spent compiling so far: 16.951722000000014\n", - "Compilation time for circuit benchmarking/ibmq/mod8-10_177.qasm: 0.4928449999999991s\n", - "Time spent compiling so far: 17.444567000000013\n", - "Compilation time for circuit benchmarking/ibmq/C17_204.qasm: 0.6808130000000006s\n", - "Time spent compiling so far: 18.125380000000014\n", - "Compilation time for circuit benchmarking/ibmq/alu-v2_31.qasm: 0.465876999999999s\n", - "Time spent compiling so far: 18.591257000000013\n", - "Compilation time for circuit benchmarking/ibmq/rd53_131.qasm: 0.6336799999999982s\n", - "Time spent compiling so far: 19.22493700000001\n", - "Compilation time for circuit benchmarking/ibmq/alu-v2_30.qasm: 0.5685000000000002s\n", - "Time spent compiling so far: 19.79343700000001\n", - "Compilation time for circuit benchmarking/ibmq/mod5adder_127.qasm: 0.5798150000000035s\n", - "Time spent compiling so far: 20.373252000000015\n", - "Compilation time for circuit benchmarking/ibmq/rd53_133.qasm: 0.7312700000000021s\n", - "Time spent compiling so far: 21.104522000000017\n", - "Compilation time for circuit benchmarking/ibmq/cm82a_208.qasm: 0.7972960000000029s\n", - "Time spent compiling so far: 21.90181800000002\n", - "Compilation time for circuit benchmarking/ibmq/majority_239.qasm: 0.7681219999999982s\n", - "Time spent compiling so far: 22.669940000000018\n", - "Compilation time for circuit benchmarking/ibmq/ex2_227.qasm: 0.7928779999999946s\n", - "Time spent compiling so far: 23.462818000000013\n", - "Compilation time for circuit benchmarking/ibmq/sf_276.qasm: 0.8383489999999938s\n", - "Time spent compiling so far: 24.301167000000007\n", - "Compilation time for circuit benchmarking/ibmq/sf_274.qasm: 0.8409819999999968s\n", - "Time spent compiling so far: 25.142149000000003\n", - "Compilation time for circuit benchmarking/ibmq/con1_216.qasm: 1.661089000000004s\n", - "Time spent compiling so far: 26.803238000000007\n", - "Compilation time for circuit benchmarking/ibmq/wim_266.qasm: 4.542246999999996s\n", - "Time spent compiling so far: 31.345485000000004\n", - "Compilation time for circuit benchmarking/ibmq/rd53_130.qasm: 1.323321s\n", - "Time spent compiling so far: 32.668806000000004\n", - "Compilation time for circuit benchmarking/ibmq/f2_232.qasm: 1.946520999999997s\n", - "Time spent compiling so far: 34.615327\n", - "Compilation time for circuit benchmarking/ibmq/cm152a_212.qasm: 3.305444999999999s\n", - "Time spent compiling so far: 37.920772\n", - "Compilation time for circuit benchmarking/ibmq/rd53_251.qasm: 1.9612180000000023s\n", - "Time spent compiling so far: 39.88199\n", - "Compilation time for circuit benchmarking/ibmq/hwb5_53.qasm: 1.4080020000000033s\n", - "Time spent compiling so far: 41.289992000000005\n", - "Compilation time for circuit benchmarking/ibmq/cm42a_207.qasm: 25.442768s\n", - "Time spent compiling so far: 66.73276000000001\n", - "Compilation time for circuit benchmarking/ibmq/pm1_249.qasm: 23.057216999999994s\n", - "Time spent compiling so far: 89.78997700000001\n", - "Compilation time for circuit benchmarking/ibmq/dc1_220.qasm: 11.941876999999991s\n", - "Time spent compiling so far: 101.731854\n", - "Compilation time for circuit benchmarking/ibmq/squar5_261.qasm: 8.14356699999999s\n", - "Time spent compiling so far: 109.87542099999999\n", - "Compilation time for circuit benchmarking/ibmq/z4_268.qasm: 11.094992999999988s\n", - "Time spent compiling so far: 120.97041399999998\n", - "Compilation time for circuit benchmarking/ibmq/sqrt8_260.qasm: 12.300466s\n", - "Time spent compiling so far: 133.27087999999998\n", - "Compilation time for circuit benchmarking/ibmq/radd_250.qasm: 26.233103999999997s\n", - "Time spent compiling so far: 159.50398399999997\n", - "Compilation time for circuit benchmarking/ibmq/adr4_197.qasm: 46.54046200000002s\n", - "Time spent compiling so far: 206.044446\n", - "Compilation time for circuit benchmarking/ibmq/sym6_145.qasm: 5.235140999999942s\n", - "Time spent compiling so far: 211.27958699999994\n", - "Compilation time for circuit benchmarking/ibmq/misex1_241.qasm: 57.77592099999998s\n", - "Time spent compiling so far: 269.0555079999999\n", - "Compilation time for circuit benchmarking/ibmq/rd73_252.qasm: 14.12543599999998s\n", - "Time spent compiling so far: 283.1809439999999\n", - "Compilation time for circuit benchmarking/ibmq/cycle10_2_110.qasm: 28.728320000000053s\n", - "Time spent compiling so far: 311.90926399999995\n", - "Compilation time for circuit benchmarking/ibmq/hwb6_56.qasm: 9.68947600000007s\n", - "Time spent compiling so far: 321.59874\n", - "Compilation time for circuit benchmarking/ibmq/square_root_7.qasm: 64.69754600000005s\n", - "Time spent compiling so far: 386.29628600000007\n", - "Compilation time for circuit benchmarking/ibmq/ham15_107.qasm: 588.933307s\n", - "Time spent compiling so far: 975.229593\n", - "Compilation time for circuit benchmarking/ibmq/dc2_222.qasm: 61.593504000000166s\n" - ] - }, - { - "name": "stdout", - "output_type": "stream", - "text": [ - "Time spent compiling so far: 1036.8230970000002\n", - "Compilation time for circuit benchmarking/ibmq/sqn_258.qasm: 28.884308000000146s\n", - "Time spent compiling so far: 1065.7074050000003\n", - "Compilation time for circuit benchmarking/ibmq/inc_237.qasm: 106.20392800000013s\n", - "Time spent compiling so far: 1171.9113330000005\n", - "Compilation time for circuit benchmarking/ibmq/cm85a_209.qasm: 89.76953500000013s\n", - "Time spent compiling so far: 1261.6808680000006\n", - "Compilation time for circuit benchmarking/ibmq/rd84_253.qasm: 66.61587499999996s\n", - "Time spent compiling so far: 1328.2967430000006\n", - "Compilation time for circuit benchmarking/ibmq/co14_215.qasm: 128.39122300000008s\n", - "Time spent compiling so far: 1456.6879660000006\n", - "Compilation time for circuit benchmarking/ibmq/root_255.qasm: 123.82956399999989s\n", - "Time spent compiling so far: 1580.5175300000005\n", - "Compilation time for circuit benchmarking/ibmq/mlp4_245.qasm: 242.6016039999995s\n", - "Time spent compiling so far: 1823.119134\n", - "Compilation time for circuit benchmarking/ibmq/urf2_277.qasm: 33.63925399999971s\n", - "Time spent compiling so far: 1856.7583879999997\n", - "Compilation time for circuit benchmarking/ibmq/sym9_148.qasm: 64.92562999999973s\n", - "Time spent compiling so far: 1921.6840179999995\n", - "Compilation time for circuit benchmarking/ibmq/life_238.qasm: 73.87741500000038s\n", - "Time spent compiling so far: 1995.5614329999999\n", - "Compilation time for circuit benchmarking/ibmq/hwb7_59.qasm: 40.89143699999977s\n", - "Time spent compiling so far: 2036.4528699999996\n", - "Compilation time for circuit benchmarking/ibmq/max46_240.qasm: 69.40681099999983s\n", - "Time spent compiling so far: 2105.8596809999995\n", - "Compilation time for circuit benchmarking/ibmq/clip_206.qasm: 284.1277840000007s\n", - "Time spent compiling so far: 2389.987465\n", - "Compilation time for circuit benchmarking/ibmq/9symml_195.qasm: 112.6734349999997s\n", - "Time spent compiling so far: 2502.6609\n", - "Compilation time for circuit benchmarking/ibmq/sym9_193.qasm: 112.30117599999949s\n", - "Time spent compiling so far: 2614.9620759999993\n", - "Compilation time for circuit benchmarking/ibmq/sao2_257.qasm: 270.47973500000035s\n", - "Time spent compiling so far: 2885.4418109999997\n", - "Compilation time for circuit benchmarking/ibmq/dist_223.qasm: 250.32260099999985s\n", - "Time spent compiling so far: 3135.7644119999995\n", - "Compilation time for circuit benchmarking/ibmq/urf5_280.qasm: 110.69195600000057s\n", - "Time spent compiling so far: 3246.456368\n", - "Compilation time for circuit benchmarking/ibmq/urf1_278.qasm: 124.18942200000038s\n", - "Time spent compiling so far: 3370.6457900000005\n", - "Compilation time for circuit benchmarking/ibmq/sym10_262.qasm: 302.3636999999999s\n", - "Time spent compiling so far: 3673.0094900000004\n", - "Compilation time for circuit benchmarking/ibmq/hwb8_113.qasm: 154.02841999999964s\n", - "Time spent compiling so far: 3827.03791\n", - "Compilation time for circuit benchmarking/ibmq/urf2_152.qasm: 151.88721200000055s\n", - "Time spent compiling so far: 3978.9251220000006\n", - "Compilation time for circuit benchmarking/ibmq/urf3_279.qasm: 370.62564999999995s\n", - "Time spent compiling so far: 4349.5507720000005\n", - "Compilation time for circuit benchmarking/ibmq/plus63mod4096_163.qasm: 830.132912s\n", - "Time spent compiling so far: 5179.6836840000005\n", - "Compilation time for circuit benchmarking/ibmq/urf5_158.qasm: 447.08807799999977s\n", - "Time spent compiling so far: 5626.771762\n", - "Compilation time for circuit benchmarking/ibmq/urf6_160.qasm: 1869.587466s\n", - "Time spent compiling so far: 7496.359228\n", - "Compilation time for circuit benchmarking/ibmq/urf1_149.qasm: 498.8691209999997s\n", - "Time spent compiling so far: 7995.228349\n", - "Compilation time for circuit benchmarking/ibmq/plus63mod8192_164.qasm: 1486.8815559999985s\n", - "Time spent compiling so far: 9482.109904999998\n", - "Compilation time for circuit benchmarking/ibmq/hwb9_119.qasm: 618.8746800000008s\n", - "Time spent compiling so far: 10100.984584999998\n", - "Compilation time for circuit benchmarking/ibmq/urf3_155.qasm: 1497.5685780000003s\n", - "Time spent compiling so far: 11598.553162999999\n", - "Compilation time for circuit benchmarking/ibmq/ground_state_estimation_10.qasm: 142.17033999999876s\n", - "Time spent compiling so far: 11740.723502999997\n", - "Compilation time for circuit benchmarking/ibmq/urf4_187.qasm: 1895.1665990000001s\n", - "Time spent compiling so far: 13635.890101999998\n", - " Size out Depth out CX count out \\\n", - "xor5_254.qasm 25 14 8 \n", - "graycode6_47.qasm 13 9 5 \n", - "ex1_226.qasm 25 14 8 \n", - "4gt11_84.qasm 50 32 18 \n", - "4mod5-v0_20.qasm 50 30 19 \n", - "ex-1_166.qasm 52 34 18 \n", - "4mod5-v1_22.qasm 54 32 20 \n", - "mod5d1_63.qasm 63 39 25 \n", - "ham3_102.qasm 53 36 19 \n", - "4gt11_83.qasm 76 47 29 \n", - "4gt11_82.qasm 95 57 36 \n", - "rd32-v0_66.qasm 75 51 27 \n", - "alu-v0_27.qasm 104 61 38 \n", - "4mod5-v1_24.qasm 89 57 37 \n", - "4mod5-v0_19.qasm 108 71 37 \n", - "mod5mils_65.qasm 92 65 34 \n", - "rd32-v1_68.qasm 75 51 27 \n", - "alu-v1_28.qasm 108 67 39 \n", - "alu-v2_33.qasm 95 58 35 \n", - "alu-v4_37.qasm 100 57 39 \n", - "alu-v3_35.qasm 100 57 39 \n", - "3_17_13.qasm 86 59 35 \n", - "alu-v1_29.qasm 103 61 38 \n", - "miller_11.qasm 138 90 50 \n", - "alu-v3_34.qasm 136 90 54 \n", - "decod24-v2_43.qasm 131 85 49 \n", - "decod24-v0_38.qasm 127 82 48 \n", - "mod5d2_64.qasm 145 93 58 \n", - "4gt13_92.qasm 169 109 63 \n", - "4gt13-v1_93.qasm 174 115 66 \n", - "4mod5-v0_18.qasm 180 119 70 \n", - "decod24-bdd_294.qasm 179 118 69 \n", - "one-two-three-v2_100.qasm 175 112 68 \n", - "one-two-three-v3_101.qasm 197 132 72 \n", - "4mod5-v1_23.qasm 187 123 74 \n", - "4mod5-bdd_287.qasm 193 124 70 \n", - "rd32_270.qasm 219 145 84 \n", - "4gt5_75.qasm 220 140 83 \n", - "alu-bdd_288.qasm 221 144 83 \n", - "alu-v0_26.qasm 226 143 83 \n", - "decod24-v1_41.qasm 216 140 83 \n", - "rd53_138.qasm 365 185 138 \n", - "4gt5_76.qasm 254 173 97 \n", - "4gt13_91.qasm 281 190 104 \n", - "cnt3-5_179.qasm 557 242 211 \n", - "qft_10.qasm 0 0 0 \n", - "4gt13_90.qasm 302 203 111 \n", - "alu-v4_36.qasm 290 187 109 \n", - "mini_alu_305.qasm 476 220 183 \n", - "ising_model_10.qasm 210 41 90 \n", - "ising_model_16.qasm 372 41 150 \n", - "ising_model_13.qasm 307 41 120 \n", - "4gt5_77.qasm 329 210 124 \n", - "sys6-v0_111.qasm 618 286 236 \n", - "one-two-three-v1_99.qasm 360 239 134 \n", - "one-two-three-v0_98.qasm 367 248 140 \n", - "decod24-v3_45.qasm 374 244 139 \n", - "4gt10-v1_81.qasm 374 250 144 \n", - "aj-e11_165.qasm 394 269 150 \n", - "4mod7-v0_94.qasm 434 290 162 \n", - "alu-v2_32.qasm 427 279 162 \n", - "rd73_140.qasm 638 296 245 \n", - "4mod7-v1_96.qasm 391 263 151 \n", - "4gt4-v0_80.qasm 450 297 172 \n", - "mod10_176.qasm 452 291 174 \n", - "0410184_169.qasm 738 349 289 \n", - "qft_16.qasm 0 0 0 \n", - "4gt12-v0_88.qasm 553 336 212 \n", - "rd84_142.qasm 1081 464 415 \n", - "rd53_311.qasm 870 457 339 \n", - "4_49_16.qasm 607 400 226 \n", - "sym9_146.qasm 936 446 358 \n", - "4gt12-v1_89.qasm 568 362 217 \n", - "4gt12-v0_87.qasm 621 392 235 \n", - "4gt4-v0_79.qasm 572 374 220 \n", - "hwb4_49.qasm 605 395 228 \n", - "sym6_316.qasm 859 460 334 \n", - "4gt12-v0_86.qasm 636 407 242 \n", - "4gt4-v0_72.qasm 657 400 254 \n", - "4gt4-v0_78.qasm 594 389 227 \n", - "mod10_171.qasm 626 413 240 \n", - "4gt4-v1_74.qasm 690 455 272 \n", - "rd53_135.qasm 846 522 324 \n", - "mini-alu_167.qasm 740 487 285 \n", - "one-two-three-v0_97.qasm 763 522 294 \n", - "ham7_104.qasm 884 546 343 \n", - "decod24-enable_126.qasm 922 593 353 \n", - "mod8-10_178.qasm 1013 676 386 \n", - "cnt3-5_180.qasm 1393 736 539 \n", - "ex3_229.qasm 1048 667 406 \n", - "4gt4-v0_73.qasm 1074 720 414 \n", - "mod8-10_177.qasm 1140 733 445 \n", - "C17_204.qasm 1377 876 536 \n", - "alu-v2_31.qasm 1159 751 441 \n", - "rd53_131.qasm 1250 763 480 \n", - "alu-v2_30.qasm 1402 906 546 \n", - "mod5adder_127.qasm 1538 957 581 \n", - "rd53_133.qasm 1592 979 612 \n", - "cm82a_208.qasm 1757 1040 681 \n", - "majority_239.qasm 1653 1026 633 \n", - "ex2_227.qasm 1725 1086 659 \n", - "sf_276.qasm 1975 1308 768 \n", - "sf_274.qasm 1993 1323 765 \n", - "con1_216.qasm 2857 1794 1121 \n", - "wim_266.qasm 2819 1678 1099 \n", - "rd53_130.qasm 2815 1753 1096 \n", - "f2_232.qasm 3427 2266 1325 \n", - "cm152a_212.qasm 3453 2186 1347 \n", - "rd53_251.qasm 3758 2368 1429 \n", - "hwb5_53.qasm 3713 2401 1432 \n", - "cm42a_207.qasm 5181 3014 2007 \n", - "pm1_249.qasm 5181 3014 2007 \n", - "dc1_220.qasm 5917 3684 2315 \n", - "squar5_261.qasm 6377 3854 2492 \n", - "z4_268.qasm 9164 5471 3563 \n", - "sqrt8_260.qasm 9106 5562 3542 \n", - "radd_250.qasm 9462 5688 3661 \n", - "adr4_197.qasm 10357 6055 4030 \n", - "sym6_145.qasm 10756 6805 4164 \n", - "misex1_241.qasm 14534 8925 5673 \n", - "rd73_252.qasm 15976 9680 6181 \n", - "cycle10_2_110.qasm 18487 11440 7252 \n", - "hwb6_56.qasm 18550 11834 7135 \n", - "square_root_7.qasm 22470 13158 8947 \n", - "ham15_107.qasm 25862 15757 9989 \n", - "dc2_222.qasm 29882 18173 11657 \n", - "sqn_258.qasm 30036 18448 11709 \n", - "inc_237.qasm 32472 19886 12766 \n", - "cm85a_209.qasm 35446 22104 13930 \n", - "rd84_253.qasm 40724 24005 15882 \n", - "co14_215.qasm 56374 30524 21896 \n", - "root_255.qasm 52615 30655 20482 \n", - "mlp4_245.qasm 59832 36288 23318 \n", - "urf2_277.qasm 66948 39434 26327 \n", - "sym9_148.qasm 62586 39228 24363 \n", - "life_238.qasm 68359 41635 26538 \n", - "hwb7_59.qasm 69520 43352 27136 \n", - "max46_240.qasm 79067 46649 30769 \n", - "clip_206.qasm 104519 61937 40950 \n", - "9symml_195.qasm 104299 63616 40686 \n", - "sym9_193.qasm 104299 63616 40686 \n", - "sao2_257.qasm 121374 69339 47336 \n", - "dist_223.qasm 118143 69156 46178 \n", - "urf5_280.qasm 154799 92851 60653 \n", - "urf1_278.qasm 176029 104865 69267 \n", - "sym10_262.qasm 200584 121977 78116 \n", - "hwb8_113.qasm 200371 124411 77790 \n", - "urf2_152.qasm 225731 144286 87475 \n", - "urf3_279.qasm 418175 246943 164373 \n", - "plus63mod4096_163.qasm 397164 243811 155060 \n", - "urf5_158.qasm 456435 286537 177850 \n", - "urf6_160.qasm 537153 314319 210332 \n", - "urf1_149.qasm 518233 318657 201568 \n", - "plus63mod8192_164.qasm 584830 359000 228069 \n", - "hwb9_119.qasm 616748 382737 239876 \n", - "urf3_155.qasm 1219715 751497 475140 \n", - "ground_state_estimation_10.qasm 15 7 3 \n", - "urf4_187.qasm 1513171 893289 586910 \n", - "\n", - " CX depth out Runtime \n", - "xor5_254.qasm 8 0.009510 \n", - "graycode6_47.qasm 5 0.005398 \n", - "ex1_226.qasm 8 0.008242 \n", - "4gt11_84.qasm 18 0.015389 \n", - "4mod5-v0_20.qasm 18 0.020801 \n", - "ex-1_166.qasm 18 0.017432 \n", - "4mod5-v1_22.qasm 19 0.016419 \n", - "mod5d1_63.qasm 22 0.024310 \n", - "ham3_102.qasm 19 0.016796 \n", - "4gt11_83.qasm 27 0.019891 \n", - "4gt11_82.qasm 32 0.024682 \n", - "rd32-v0_66.qasm 27 0.022649 \n", - "alu-v0_27.qasm 33 0.031051 \n", - "4mod5-v1_24.qasm 35 0.035731 \n", - "4mod5-v0_19.qasm 37 0.031680 \n", - "mod5mils_65.qasm 34 0.033256 \n", - "rd32-v1_68.qasm 27 0.031518 \n", - "alu-v1_28.qasm 35 0.044897 \n", - "alu-v2_33.qasm 31 0.035871 \n", - "alu-v4_37.qasm 34 0.032545 \n", - "alu-v3_35.qasm 34 0.031094 \n", - "3_17_13.qasm 35 0.038106 \n", - "alu-v1_29.qasm 33 0.033971 \n", - "miller_11.qasm 50 0.053864 \n", - "alu-v3_34.qasm 53 0.063358 \n", - "decod24-v2_43.qasm 49 0.055582 \n", - "decod24-v0_38.qasm 48 0.053290 \n", - "mod5d2_64.qasm 56 0.043774 \n", - "4gt13_92.qasm 58 0.073271 \n", - "4gt13-v1_93.qasm 62 0.068343 \n", - "4mod5-v0_18.qasm 70 0.054039 \n", - "decod24-bdd_294.qasm 68 0.070724 \n", - "one-two-three-v2_100.qasm 64 0.074335 \n", - "one-two-three-v3_101.qasm 70 0.065476 \n", - "4mod5-v1_23.qasm 72 0.053468 \n", - "4mod5-bdd_287.qasm 69 0.078521 \n", - "rd32_270.qasm 82 0.061950 \n", - "4gt5_75.qasm 78 0.081763 \n", - "alu-bdd_288.qasm 80 0.064100 \n", - "alu-v0_26.qasm 78 0.081182 \n", - "decod24-v1_41.qasm 79 0.081245 \n", - "rd53_138.qasm 103 0.133514 \n", - "4gt5_76.qasm 93 0.068765 \n", - "4gt13_91.qasm 101 0.082594 \n", - "cnt3-5_179.qasm 134 0.281470 \n", - "qft_10.qasm 0 0.000000 \n", - "4gt13_90.qasm 108 0.092554 \n", - "alu-v4_36.qasm 104 0.117040 \n", - "mini_alu_305.qasm 125 0.212324 \n", - "ising_model_10.qasm 20 0.261262 \n", - "ising_model_16.qasm 20 0.376977 \n", - "ising_model_13.qasm 20 0.334911 \n", - "4gt5_77.qasm 117 0.129335 \n", - "sys6-v0_111.qasm 159 0.304467 \n", - "one-two-three-v1_99.qasm 131 0.153367 \n", - "one-two-three-v0_98.qasm 135 0.126284 \n", - "decod24-v3_45.qasm 133 0.149087 \n", - "4gt10-v1_81.qasm 140 0.140766 \n", - "aj-e11_165.qasm 145 0.149700 \n", - "4mod7-v0_94.qasm 158 0.163860 \n", - "alu-v2_32.qasm 155 0.159586 \n", - "rd73_140.qasm 166 0.263072 \n", - "4mod7-v1_96.qasm 145 0.159860 \n", - "4gt4-v0_80.qasm 163 0.182525 \n", - "mod10_176.qasm 163 0.179933 \n", - "0410184_169.qasm 196 0.624469 \n", - "qft_16.qasm 0 0.000000 \n", - "4gt12-v0_88.qasm 188 0.208980 \n", - "rd84_142.qasm 258 0.740196 \n", - "rd53_311.qasm 254 0.502639 \n", - "4_49_16.qasm 218 0.228873 \n", - "sym9_146.qasm 252 0.566968 \n", - "4gt12-v1_89.qasm 203 0.246257 \n", - "4gt12-v0_87.qasm 215 0.218950 \n", - "4gt4-v0_79.qasm 208 0.195003 \n", - "hwb4_49.qasm 217 0.236325 \n", - "sym6_316.qasm 257 0.657216 \n", - "4gt12-v0_86.qasm 223 0.240528 \n", - "4gt4-v0_72.qasm 227 0.283360 \n", - "4gt4-v0_78.qasm 216 0.200468 \n", - "mod10_171.qasm 228 0.237878 \n", - "4gt4-v1_74.qasm 261 0.289275 \n", - "rd53_135.qasm 291 0.379021 \n", - "mini-alu_167.qasm 267 0.289529 \n", - "one-two-three-v0_97.qasm 289 0.324230 \n", - "ham7_104.qasm 312 0.374694 \n", - "decod24-enable_126.qasm 333 0.373649 \n", - "mod8-10_178.qasm 367 0.431755 \n", - "cnt3-5_180.qasm 410 2.258793 \n", - "ex3_229.qasm 372 0.423773 \n", - "4gt4-v0_73.qasm 394 0.436116 \n", - "mod8-10_177.qasm 416 0.492845 \n", - "C17_204.qasm 486 0.680813 \n", - "alu-v2_31.qasm 415 0.465877 \n", - "rd53_131.qasm 432 0.633680 \n", - "alu-v2_30.qasm 504 0.568500 \n", - "mod5adder_127.qasm 536 0.579815 \n", - "rd53_133.qasm 553 0.731270 \n", - "cm82a_208.qasm 579 0.797296 \n", - "majority_239.qasm 565 0.768122 \n", - "ex2_227.qasm 604 0.792878 \n", - "sf_276.qasm 734 0.838349 \n", - "sf_274.qasm 732 0.840982 \n", - "con1_216.qasm 993 1.661089 \n", - "wim_266.qasm 934 4.542247 \n", - "rd53_130.qasm 980 1.323321 \n", - "f2_232.qasm 1237 1.946521 \n", - "cm152a_212.qasm 1225 3.305445 \n", - "rd53_251.qasm 1290 1.961218 \n", - "hwb5_53.qasm 1352 1.408002 \n", - "cm42a_207.qasm 1694 25.442768 \n", - "pm1_249.qasm 1694 23.057217 \n", - "dc1_220.qasm 2045 11.941877 \n", - "squar5_261.qasm 2161 8.143567 \n", - "z4_268.qasm 3030 11.094993 \n", - "sqrt8_260.qasm 3093 12.300466 \n", - "radd_250.qasm 3167 26.233104 \n", - "adr4_197.qasm 3382 46.540462 \n", - "sym6_145.qasm 3819 5.235141 \n", - "misex1_241.qasm 4974 57.775921 \n", - "rd73_252.qasm 5373 14.125436 \n", - "cycle10_2_110.qasm 6397 28.728320 \n", - "hwb6_56.qasm 6513 9.689476 \n", - "square_root_7.qasm 7446 64.697546 \n", - "ham15_107.qasm 8728 588.933307 \n", - "dc2_222.qasm 10098 61.593504 \n", - "sqn_258.qasm 10273 28.884308 \n", - "inc_237.qasm 11140 106.203928 \n", - "cm85a_209.qasm 12366 89.769535 \n", - "rd84_253.qasm 13405 66.615875 \n", - "co14_215.qasm 17008 128.391223 \n", - "root_255.qasm 17076 123.829564 \n", - "mlp4_245.qasm 20258 242.601604 \n", - "urf2_277.qasm 22099 33.639254 \n", - "sym9_148.qasm 21805 64.925630 \n", - "life_238.qasm 23121 73.877415 \n", - "hwb7_59.qasm 24091 40.891437 \n", - "max46_240.qasm 26016 69.406811 \n", - "clip_206.qasm 34520 284.127784 \n", - "9symml_195.qasm 35437 112.673435 \n", - "sym9_193.qasm 35437 112.301176 \n", - "sao2_257.qasm 38642 270.479735 \n", - "dist_223.qasm 38660 250.322601 \n", - "urf5_280.qasm 51913 110.691956 \n", - "urf1_278.qasm 58728 124.189422 \n", - "sym10_262.qasm 68141 302.363700 \n", - "hwb8_113.qasm 69065 154.028420 \n", - "urf2_152.qasm 79903 151.887212 \n", - "urf3_279.qasm 138114 370.625650 \n", - "plus63mod4096_163.qasm 136210 830.132912 \n", - "urf5_158.qasm 159416 447.088078 \n", - "urf6_160.qasm 175227 1869.587466 \n", - "urf1_149.qasm 179176 498.869121 \n", - "plus63mod8192_164.qasm 200254 1486.881556 \n", - "hwb9_119.qasm 211857 618.874680 \n", - "urf3_155.qasm 418559 1497.568578 \n", - "ground_state_estimation_10.qasm 3 142.170340 \n", - "urf4_187.qasm 494739 1895.166599 \n" - ] - } - ], - "source": [ - "test_table = pandas.read_csv(\"benchmarking/IBMQConfig.csv\",index_col=0)\n", - "test_table = test_table.sort_values(by='Depth in')\n", - "\n", - "stat_table = pandas.DataFrame({})\n", - "total_time = 0\n", - "\n", - "for i, (index, row) in enumerate(test_table.iterrows()):\n", - " filename = row['Filename']\n", - " new_stats = getStats(filename, directed_arc)\n", - " total_time += new_stats[4] ###\n", - " print(\"Time spent compiling so far: \" + str(total_time))\n", - " new_table_row = pandas.DataFrame.from_dict({index : new_stats}, \n", - " orient='index', columns=['Size out', 'Depth out', 'CX count out', 'CX depth out','Runtime'])\n", - " stat_table = stat_table.append(new_table_row)\n", - "\n", - "#stat_table.to_csv(\"BenchmarkTket.csv\") ###Note: uncomment this line to print table to csv.\n", - "with pandas.option_context('display.max_rows', None):\n", - " print(stat_table)" - ] - } - ], - "metadata": { - "kernelspec": { - "display_name": "Python [conda env:tket] *", - "language": "python", - "name": "conda-env-tket-py" - }, - "language_info": { - "codemirror_mode": { - "name": "ipython", - "version": 3 - }, - "file_extension": ".py", - "mimetype": "text/x-python", - "name": "python", - "nbconvert_exporter": "python", - "pygments_lexer": "ipython3", - "version": "3.6.8" - } - }, - "nbformat": 4, - "nbformat_minor": 2 -} From ca9f0bcf04bfb7d9f6f55f23f8b4c654b6eaad6e Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Tue, 7 Nov 2023 11:34:28 +0000 Subject: [PATCH 36/51] clean up notebook --- examples/phase_estimation.ipynb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/examples/phase_estimation.ipynb b/examples/phase_estimation.ipynb index f68b02f8..a3c2455d 100644 --- a/examples/phase_estimation.ipynb +++ b/examples/phase_estimation.ipynb @@ -1 +1 @@ -{"cells":[{"cell_type":"markdown","metadata":{},"source":["# Quantum Phase Estimation\n","\n","When constructing circuits for quantum algorithms it is useful to think of higher level operations than just individual quantum gates.\n","\n","In `pytket` we can construct circuits using box structures which abstract away the complexity of the underlying circuit.\n","\n","This notebook is intended to complement the [boxes section](https://tket.quantinuum.com/user-manual/manual_circuit.html#boxes) of the user manual which introduces the different box types.\n","\n","To demonstrate boxes in `pytket` we will consider the Quantum Phase Estimation algorithm (QPE). This is an important subroutine in several quantum algorithms including Shor's algorithm and fault-tolerant approaches to quantum chemistry.\n","\n","## Overview of Phase Estimation\n","\n","The Quantum Phase Estimation algorithm can be used to estimate the eigenvalues of some unitary operator $U$ to some desired precision.\n","\n","The eigenvalues of $U$ lie on the unit circle, giving us the following eigenvalue equation\n","\n","$$\n","\\begin{equation}\n","U |\\psi \\rangle = e^{2 \\pi i \\theta} |\\psi\\rangle\\,, \\quad 0 \\leq \\theta \\leq 1\n","\\end{equation}\n","$$\n","\n","Here $|\\psi \\rangle$ is an eigenstate of the operator $U$. In phase estimation we estimate the eigenvalue $e^{2 \\pi i \\theta}$ by approximating $\\theta$.\n","\n","\n","The circuit for Quantum phase estimation is itself composed of several subroutines which we can realise as boxes.\n","\n","![](images/phase_est.png \"Quantum Phase Estimation Circuit\")"]},{"cell_type":"markdown","metadata":{},"source":["QPE is generally split up into three stages\n","\n","1. Firstly we prepare an initial state in one register. In parallel we prepare a uniform superposition state using Hadamard gates on some ancilla qubits. The number of ancilla qubits determines how precisely we can estimate the phase $\\theta$.\n","\n","2. Secondly we apply successive controlled $U$ gates. This has the effect of \"kicking back\" phases onto the ancilla qubits according to the eigenvalue equation above.\n","\n","3. Finally we apply the inverse Quantum Fourier Transform (QFT). This essentially plays the role of destructive interference, suppressing amplitudes from \"undesirable states\" and hopefully allowing us to measure a single outcome (or a small number of outcomes) with high probability.\n","\n","\n","There is some subtlety around the first point. The initial state used can be an exact eigenstate of $U$ however this may be difficult to prepare if we don't know the eigenvalues of $U$ in advance. Alternatively we could use an initial state that is a linear combination of eigenstates, as the phase estimation will project into the eigenspace of $U$."]},{"cell_type":"markdown","metadata":{},"source":["We also assume that we can implement $U$ with a quantum circuit. In chemistry applications $U$ could be of the form $U=e^{-iHt}$ where $H$ is the Hamiltonian of some system of interest. In the cannonical algorithm, the number of controlled unitaries we apply scales exponentially with the number of ancilla qubits. This allows more precision at the expense of a larger quantum circuit."]},{"cell_type":"markdown","metadata":{},"source":["## The Quantum Fourier Transform"]},{"cell_type":"markdown","metadata":{},"source":["Before considering the other parts of the QPE algorithm, lets focus on the Quantum Fourier Transform (QFT) subroutine.\n","\n","Mathematically, the QFT has the following action.\n","\n","$$\n","\\begin{equation}\n","QFT : |j\\rangle\\ \\longmapsto \\sum_{k=0}^{N - 1} e^{2 \\pi ijk/N}|k\\rangle, \\quad N= 2^k\n","\\end{equation}\n","$$\n","\n","This is essentially the Discrete Fourier transform except the input is a quantum state $|j\\rangle$.\n","\n","It is well known that the QFT can be implemented efficiently with a quantum circuit\n","\n","We can build the circuit for the $n$ qubit QFT using $n$ Hadamard gates $\\frac{n}{2}$ swap gates and $\\frac{n(n-1)}{2}$ controlled unitary rotations $\\text{CU1}$.\n","\n","$$\n"," \\begin{equation}\n"," CU1(\\phi) =\n"," \\begin{pmatrix}\n"," I & 0 \\\\\n"," 0 & U1(\\phi)\n"," \\end{pmatrix}\n"," \\,, \\quad\n","U1(\\phi) =\n"," \\begin{pmatrix}\n"," 1 & 0 \\\\\n"," 0 & e^{i \\phi}\n"," \\end{pmatrix}\n"," \\end{equation}\n","$$\n","\n","The circuit for the Quantum Fourier transform on three qubits is the following\n","\n","![](images/qft.png \"QFT Circuit\")\n","\n","We can build this circuit in `pytket` by adding gate operations manually:"]},{"cell_type":"markdown","metadata":{},"source":["lets build the QFT for three qubits"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import Circuit\n","from pytket.circuit.display import render_circuit_jupyter"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft3_circ = Circuit(3)\n","qft3_circ.H(0)\n","qft3_circ.CU1(0.5, 1, 0)\n","qft3_circ.CU1(0.25, 2, 0)\n","qft3_circ.H(1)\n","qft3_circ.CU1(0.5, 2, 1)\n","qft3_circ.H(2)\n","qft3_circ.SWAP(0, 2)\n","render_circuit_jupyter(qft3_circ)"]},{"cell_type":"markdown","metadata":{},"source":["We can generalise the quantum Fourier transform to $n$ qubits by iterating over the qubits as follows"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def build_qft_circuit(n_qubits: int) -> Circuit:\n"," circ = Circuit(n_qubits, name=\"QFT\")\n"," for i in range(n_qubits):\n"," circ.H(i)\n"," for j in range(i + 1, n_qubits):\n"," circ.CU1(1 / 2 ** (j - i), j, i)\n"," for k in range(0, n_qubits // 2):\n"," circ.SWAP(k, n_qubits - k - 1)\n"," return circ"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft4_circ: Circuit = build_qft_circuit(4)\n","render_circuit_jupyter(qft4_circ)"]},{"cell_type":"markdown","metadata":{},"source":["Now that we have the generalised circuit we can wrap it up in a `CircBox` which can then be added to another circuit as a subroutine."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import CircBox"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft4_box: CircBox = CircBox(qft4_circ)\n","qft_circ = Circuit(4).add_gate(qft4_box, [0, 1, 2, 3])\n","render_circuit_jupyter(qft_circ)"]},{"cell_type":"markdown","metadata":{},"source":["Note how the `CircBox` inherits the name `QFT` from the underlying circuit."]},{"cell_type":"markdown","metadata":{},"source":["Recall that in our phase estimation algorithm we need to use the inverse QFT.\n","\n","$$\n","\\begin{equation}\n","\\text{QFT}^† : \\sum_{k=0}^{N - 1} e^{2 \\pi ijk/N}|k\\rangle \\longmapsto |j\\rangle\\,, \\quad N= 2^k\n","\\end{equation}\n","$$\n","\n","\n","Now that we have the QFT circuit we can obtain the inverse by using `CircBox.dagger`. We can also verify that this is correct by inspecting the circuit inside with `CircBox.get_circuit()`."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["inv_qft4_box = qft4_box.dagger\n","render_circuit_jupyter(inv_qft4_box.get_circuit())"]},{"cell_type":"markdown","metadata":{},"source":["## The Controlled Unitary Operations"]},{"cell_type":"markdown","metadata":{},"source":["In the phase estimation algorithm we repeatedly perform controlled unitary operations. In the canonical variant, the number of controlled unitaries will be $2^m - 1$ where $m$ is the number of measurement qubits."]},{"cell_type":"markdown","metadata":{},"source":["The form of $U$ will vary depending on the application. For chemistry or condensed matter physics $U$ typically be the time evolution operator $U(t) = e^{- i H t}$ where $H$ is the problem Hamiltonian."]},{"cell_type":"markdown","metadata":{},"source":["Suppose that we had the following decomposition for $H$ in terms of Pauli strings $P_j$ and complex coefficients $\\alpha_j$.\n","\n","$$\n","\\begin{equation}\n","H = \\sum_j \\alpha_j P_j\\,, \\quad \\, P_j \\in \\{I, \\,X, \\,Y, \\,Z\\}^{\\otimes n}\n","\\end{equation}\n","$$\n","\n","Here Pauli strings refers to tensor products of Pauli operators. These strings form an orthonormal basis for $2^n \\times 2^n$ matrices."]},{"cell_type":"markdown","metadata":{},"source":["If we have a Hamiltonian in the form above, we can then implement $U(t)$ as a sequence of Pauli gadget circuits. We can do this with the [PauliExpBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.PauliExpBox) construct in pytket. For more on `PauliExpBox` see the [user manual](https://tket.quantinuum.com/user-manual/manual_circuit.html#pauli-exponential-boxes)."]},{"cell_type":"markdown","metadata":{},"source":["Once we have a circuit to implement our time evolution operator $U(t)$, we can construct the controlled $U(t)$ operations using [QControlBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.QControlBox). If our base unitary is a sequence of `PauliExpBox`(es) then there is some structure we can exploit to simplify our circuit. See this [blog post](https://tket.quantinuum.com/tket-blog/posts/controlled_gates/) on [ConjugationBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.ConjugationBox) for more."]},{"cell_type":"markdown","metadata":{},"source":["In what follows, we will just construct a simplified instance of QPE where the controlled unitaries are just $\\text{CU1}$ gates."]},{"cell_type":"markdown","metadata":{},"source":["## Putting it all together"]},{"cell_type":"markdown","metadata":{},"source":["We can now define a function to build our entire QPE circuit. We can make this function take a state preparation circuit and a unitary circuit as input as well. The function also has the number of measurement qubits as input which will determine the precision of our phase estimate."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import QControlBox"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def build_phase_est_circuit(\n"," n_measurement_qubits: int, state_prep_circuit: Circuit, unitary_circuit: Circuit\n",") -> Circuit:\n"," qpe_circ: Circuit = Circuit()\n"," n_state_prep_qubits = state_prep_circuit.n_qubits\n"," measurement_register = qpe_circ.add_q_register(\"m\", n_measurement_qubits)\n"," state_prep_register = qpe_circ.add_q_register(\"p\", n_state_prep_qubits)\n"," qpe_circ.add_circuit(state_prep_circuit, list(state_prep_register))\n","\n"," # Create a controlled unitary with a single control qubit\n"," unitary_circuit.name = \"U\"\n"," controlled_u_gate = QControlBox(CircBox(unitary_circuit), 1)\n","\n"," # Add Hadamard gates to every qubit in the measurement register\n"," for m_qubit in measurement_register:\n"," qpe_circ.H(m_qubit)\n","\n"," # Add all (2**n_measurement_qubits - 1) of the controlled unitaries sequentially\n"," for m_qubit in range(n_measurement_qubits):\n"," control_index = n_measurement_qubits - m_qubit - 1\n"," control_qubit = [measurement_register[control_index]]\n"," for _ in range(2**m_qubit):\n"," qpe_circ.add_qcontrolbox(\n"," controlled_u_gate, control_qubit + list(state_prep_register)\n"," )\n","\n"," # Finally, append the inverse qft and measure the qubits\n"," qft_box = CircBox(build_qft_circuit(n_measurement_qubits))\n"," inverse_qft_box = qft_box.dagger\n"," qpe_circ.add_circbox(inverse_qft_box, list(measurement_register))\n"," qpe_circ.measure_register(measurement_register, \"c\")\n"," return qpe_circ"]},{"cell_type":"markdown","metadata":{},"source":["## Phase Estimation with a Trivial Eigenstate\n","\n","Lets test our circuit construction by preparing a trivial $|1\\rangle$ eigenstate of the $\\text{U1}$ gate. We can then see if our phase estimation circuit returns the expected eigenvalue."]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","U1(\\phi)|1\\rangle = e^{i\\phi} = e^{2 \\pi i \\theta} \\implies \\theta = \\frac{\\phi}{2}\n","\\end{equation}\n","$$\n","\n","So we expect that our ideal phase $\\theta$ will be half the input angle $\\phi$ to our $U1$ gate."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["prep_circuit = Circuit(1).X(0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["input_angle = 0.73 # angle as number of half turns"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["unitary_circuit = Circuit(1).U1(input_angle, 0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qpe_circ_trivial = build_phase_est_circuit(\n"," 4, state_prep_circuit=prep_circuit, unitary_circuit=unitary_circuit\n",")"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["render_circuit_jupyter(qpe_circ_trivial)"]},{"cell_type":"markdown","metadata":{},"source":["Lets use the noiseless `AerBackend` simulator to run our phase estimation circuit."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.extensions.qiskit import AerBackend"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["backend = AerBackend()"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["compiled_circ = backend.get_compiled_circuit(qpe_circ_trivial)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_shots = 1000\n","result = backend.run_circuit(compiled_circ, n_shots)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(result.get_counts())"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.backends.backendresult import BackendResult\n","import matplotlib.pyplot as plt"]},{"cell_type":"markdown","metadata":{},"source":["plotting function for QPE Notebook"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def plot_qpe_results(\n"," sim_result: BackendResult,\n"," n_strings: int = 4,\n"," dark_mode: bool = False,\n"," y_limit: int = 1000,\n",") -> None:\n"," \"\"\"\n"," Plots results in a barchart given a BackendResult. the number of stings displayed\n"," can be specified with the n_strings argument.\n"," \"\"\"\n"," counts_dict = sim_result.get_counts()\n"," sorted_shots = counts_dict.most_common()\n"," n_most_common_strings = sorted_shots[:n_strings]\n"," x_axis_values = [str(entry[0]) for entry in n_most_common_strings] # basis states\n"," y_axis_values = [entry[1] for entry in n_most_common_strings] # counts\n"," if dark_mode:\n"," plt.style.use(\"dark_background\")\n"," fig = plt.figure()\n"," ax = fig.add_axes((0, 0, 0.75, 0.5))\n"," color_list = [\"orange\"] * (len(x_axis_values))\n"," ax.bar(\n"," x=x_axis_values,\n"," height=y_axis_values,\n"," color=color_list,\n"," )\n"," ax.set_title(label=\"Results\")\n"," plt.ylim([0, y_limit])\n"," plt.xlabel(\"Basis State\")\n"," plt.ylabel(\"Number of Shots\")\n"," plt.show()"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plot_qpe_results(result, y_limit=int(1.2 * n_shots))"]},{"cell_type":"markdown","metadata":{},"source":["As expected we see one outcome with high probability. Lets now extract our approximation of $\\theta$ from our output bitstrings.\n","\n","suppose the $j$ is an integer representation of our most commonly measured bitstring."]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","\\theta_{estimate} = \\frac{j}{N}\n","\\end{equation}\n","$$"]},{"cell_type":"markdown","metadata":{},"source":["Here $N = 2 ^n$ where $n$ is the number of measurement qubits."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.backends.backendresult import BackendResult"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def single_phase_from_backendresult(result: BackendResult) -> float:\n"," # Extract most common measurement outcome\n"," basis_state = result.get_counts().most_common()[0][0]\n"," bitstring = \"\".join([str(bit) for bit in basis_state])\n"," integer = int(bitstring, 2)\n","\n"," # Calculate theta estimate\n"," return integer / (2 ** len(bitstring))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["theta = single_phase_from_backendresult(result)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(theta)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(input_angle / 2)"]},{"cell_type":"markdown","metadata":{},"source":["Our output is close to half our input angle $\\phi$ as expected. Lets calculate our error."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["error = round(abs(input_angle - (2 * theta)), 3)\n","print(error)"]},{"cell_type":"markdown","metadata":{},"source":["## Suggestions for further reading\n","\n","In this notebook we have shown the canonical variant of quantum phase estimation. There are several other variants.\n","\n","Quantinuum paper on Bayesian phase estimation -> https://arxiv.org/pdf/2306.16608.pdf\n","Blog post on `ConjugationBox` -> https://tket.quantinuum.com/tket-blog/posts/controlled_gates/ - efficient circuits for controlled Pauli gadgets.\n","\n","As mentioned quantum phase estimation is a subroutine in Shor's algorithm. Read more about how phase estimation is used in period finding."]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} +{"cells":[{"cell_type":"markdown","metadata":{},"source":["# Quantum Phase Estimation\n","\n","When constructing circuits for quantum algorithms it is useful to think of higher level operations than just individual quantum gates.\n","\n","In `pytket` we can construct circuits using box structures which abstract away the complexity of the underlying circuit.\n","\n","This notebook is intended to complement the [boxes section](https://tket.quantinuum.com/user-manual/manual_circuit.html#boxes) of the user manual which introduces the different box types.\n","\n","To demonstrate boxes in `pytket` we will consider the Quantum Phase Estimation algorithm (QPE). This is an important subroutine in several quantum algorithms including Shor's algorithm and fault-tolerant approaches to quantum chemistry.\n","\n","## Overview of Phase Estimation\n","\n","The Quantum Phase Estimation algorithm can be used to estimate the eigenvalues of some unitary operator $U$ to some desired precision.\n","\n","The eigenvalues of $U$ lie on the unit circle, giving us the following eigenvalue equation\n","\n","$$\n","\\begin{equation}\n","U |\\psi \\rangle = e^{2 \\pi i \\theta} |\\psi\\rangle\\,, \\quad 0 \\leq \\theta \\leq 1\n","\\end{equation}\n","$$\n","\n","Here $|\\psi \\rangle$ is an eigenstate of the operator $U$. In phase estimation we estimate the eigenvalue $e^{2 \\pi i \\theta}$ by approximating $\\theta$.\n","\n","\n","The circuit for Quantum phase estimation is itself composed of several subroutines which we can realise as boxes.\n","\n","![](images/phase_est.png \"Quantum Phase Estimation Circuit\")"]},{"cell_type":"markdown","metadata":{},"source":["QPE is generally split up into three stages
\n","
\n","1. Firstly we prepare an initial state in one register. In parallel we prepare a uniform superposition state using Hadamard gates on some ancilla qubits. The number of ancilla qubits determines how precisely we can estimate the phase $\\theta$.
\n","
\n","2. Secondly we apply successive controlled $U$ gates. This has the effect of \"kicking back\" phases onto the ancilla qubits according to the eigenvalue equation above.
\n","
\n","3. Finally we apply the inverse Quantum Fourier Transform (QFT). This essentially plays the role of destructive interference, suppressing amplitudes from \"undesirable states\" and hopefully allowing us to measure a single outcome (or a small number of outcomes) with high probability.
\n","
\n","
\n","There is some subtlety around the first point. The initial state used can be an exact eigenstate of $U$ however this may be difficult to prepare if we don't know the eigenvalues of $U$ in advance. Alternatively we could use an initial state that is a linear combination of eigenstates, as the phase estimation will project into the eigenspace of $U$."]},{"cell_type":"markdown","metadata":{},"source":["We also assume that we can implement $U$ with a quantum circuit. In chemistry applications $U$ could be of the form $U=e^{-iHt}$ where $H$ is the Hamiltonian of some system of interest. In the cannonical algorithm, the number of controlled unitaries we apply scales exponentially with the number of ancilla qubits. This allows more precision at the expense of a larger quantum circuit."]},{"cell_type":"markdown","metadata":{},"source":["## The Quantum Fourier Transform"]},{"cell_type":"markdown","metadata":{},"source":["Before considering the other parts of the QPE algorithm, lets focus on the Quantum Fourier Transform (QFT) subroutine.\n","\n","Mathematically, the QFT has the following action.\n","\n","$$\n","\\begin{equation}\n","QFT : |j\\rangle\\ \\longmapsto \\sum_{k=0}^{N - 1} e^{2 \\pi ijk/N}|k\\rangle, \\quad N= 2^k\n","\\end{equation}\n","$$\n","\n","This is essentially the Discrete Fourier transform except the input is a quantum state $|j\\rangle$.\n","\n","It is well known that the QFT can be implemented efficiently with a quantum circuit\n","\n","We can build the circuit for the $n$ qubit QFT using $n$ Hadamard gates $\\frac{n}{2}$ swap gates and $\\frac{n(n-1)}{2}$ controlled unitary rotations $\\text{CU1}$.\n","\n","$$\n"," \\begin{equation}\n"," CU1(\\phi) =\n"," \\begin{pmatrix}\n"," I & 0 \\\\\n"," 0 & U1(\\phi)\n"," \\end{pmatrix}\n"," \\,, \\quad\n","U1(\\phi) =\n"," \\begin{pmatrix}\n"," 1 & 0 \\\\\n"," 0 & e^{i \\phi}\n"," \\end{pmatrix}\n"," \\end{equation}\n","$$\n","\n","The circuit for the Quantum Fourier transform on three qubits is the following\n","\n","![](images/qft.png \"QFT Circuit\")\n","\n","We can build this circuit in `pytket` by adding gate operations manually:"]},{"cell_type":"markdown","metadata":{},"source":["lets build the QFT for three qubits"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import Circuit\n","from pytket.circuit.display import render_circuit_jupyter"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft3_circ = Circuit(3)\n","qft3_circ.H(0)\n","qft3_circ.CU1(0.5, 1, 0)\n","qft3_circ.CU1(0.25, 2, 0)\n","qft3_circ.H(1)\n","qft3_circ.CU1(0.5, 2, 1)\n","qft3_circ.H(2)\n","qft3_circ.SWAP(0, 2)\n","render_circuit_jupyter(qft3_circ)"]},{"cell_type":"markdown","metadata":{},"source":["We can generalise the quantum Fourier transform to $n$ qubits by iterating over the qubits as follows"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def build_qft_circuit(n_qubits: int) -> Circuit:\n"," circ = Circuit(n_qubits, name=\"QFT\")\n"," for i in range(n_qubits):\n"," circ.H(i)\n"," for j in range(i + 1, n_qubits):\n"," circ.CU1(1 / 2 ** (j - i), j, i)\n"," for k in range(0, n_qubits // 2):\n"," circ.SWAP(k, n_qubits - k - 1)\n"," return circ"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft4_circ: Circuit = build_qft_circuit(4)\n","render_circuit_jupyter(qft4_circ)"]},{"cell_type":"markdown","metadata":{},"source":["Now that we have the generalised circuit we can wrap it up in a `CircBox` which can then be added to another circuit as a subroutine."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import CircBox"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft4_box: CircBox = CircBox(qft4_circ)\n","qft_circ = Circuit(4).add_gate(qft4_box, [0, 1, 2, 3])\n","render_circuit_jupyter(qft_circ)"]},{"cell_type":"markdown","metadata":{},"source":["Note how the `CircBox` inherits the name `QFT` from the underlying circuit."]},{"cell_type":"markdown","metadata":{},"source":["Recall that in our phase estimation algorithm we need to use the inverse QFT.\n","\n","$$\n","\\begin{equation}\n","\\text{QFT}^† : \\sum_{k=0}^{N - 1} e^{2 \\pi ijk/N}|k\\rangle \\longmapsto |j\\rangle\\,, \\quad N= 2^k\n","\\end{equation}\n","$$\n","\n","\n","Now that we have the QFT circuit we can obtain the inverse by using `CircBox.dagger`. We can also verify that this is correct by inspecting the circuit inside with `CircBox.get_circuit()`."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["inv_qft4_box = qft4_box.dagger\n","render_circuit_jupyter(inv_qft4_box.get_circuit())"]},{"cell_type":"markdown","metadata":{},"source":["## The Controlled Unitary Operations"]},{"cell_type":"markdown","metadata":{},"source":["In the phase estimation algorithm we repeatedly perform controlled unitary operations. In the canonical variant, the number of controlled unitaries will be $2^m - 1$ where $m$ is the number of measurement qubits."]},{"cell_type":"markdown","metadata":{},"source":["The form of $U$ will vary depending on the application. For chemistry or condensed matter physics $U$ typically be the time evolution operator $U(t) = e^{- i H t}$ where $H$ is the problem Hamiltonian."]},{"cell_type":"markdown","metadata":{},"source":["Suppose that we had the following decomposition for $H$ in terms of Pauli strings $P_j$ and complex coefficients $\\alpha_j$.\n","\n","$$\n","\\begin{equation}\n","H = \\sum_j \\alpha_j P_j\\,, \\quad \\, P_j \\in \\{I, \\,X, \\,Y, \\,Z\\}^{\\otimes n}\n","\\end{equation}\n","$$\n","\n","Here Pauli strings refers to tensor products of Pauli operators. These strings form an orthonormal basis for $2^n \\times 2^n$ matrices."]},{"cell_type":"markdown","metadata":{},"source":["If we have a Hamiltonian in the form above, we can then implement $U(t)$ as a sequence of Pauli gadget circuits. We can do this with the [PauliExpBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.PauliExpBox) construct in pytket. For more on `PauliExpBox` see the [user manual](https://tket.quantinuum.com/user-manual/manual_circuit.html#pauli-exponential-boxes)."]},{"cell_type":"markdown","metadata":{},"source":["Once we have a circuit to implement our time evolution operator $U(t)$, we can construct the controlled $U(t)$ operations using [QControlBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.QControlBox). If our base unitary is a sequence of `PauliExpBox`(es) then there is some structure we can exploit to simplify our circuit. See this [blog post](https://tket.quantinuum.com/tket-blog/posts/controlled_gates/) on [ConjugationBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.ConjugationBox) for more."]},{"cell_type":"markdown","metadata":{},"source":["In what follows, we will just construct a simplified instance of QPE where the controlled unitaries are just $\\text{CU1}$ gates."]},{"cell_type":"markdown","metadata":{},"source":["## Putting it all together"]},{"cell_type":"markdown","metadata":{},"source":["We can now define a function to build our entire QPE circuit. We can make this function take a state preparation circuit and a unitary circuit as input as well. The function also has the number of measurement qubits as input which will determine the precision of our phase estimate."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import QControlBox"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def build_phase_est_circuit(\n"," n_measurement_qubits: int, state_prep_circuit: Circuit, unitary_circuit: Circuit\n",") -> Circuit:\n"," qpe_circ: Circuit = Circuit()\n"," n_state_prep_qubits = state_prep_circuit.n_qubits\n"," measurement_register = qpe_circ.add_q_register(\"m\", n_measurement_qubits)\n"," state_prep_register = qpe_circ.add_q_register(\"p\", n_state_prep_qubits)\n"," qpe_circ.add_circuit(state_prep_circuit, list(state_prep_register))\n","\n"," # Create a controlled unitary with a single control qubit\n"," unitary_circuit.name = \"U\"\n"," controlled_u_gate = QControlBox(CircBox(unitary_circuit), 1)\n","\n"," # Add Hadamard gates to every qubit in the measurement register\n"," for m_qubit in measurement_register:\n"," qpe_circ.H(m_qubit)\n","\n"," # Add all (2**n_measurement_qubits - 1) of the controlled unitaries sequentially\n"," for m_qubit in range(n_measurement_qubits):\n"," control_index = n_measurement_qubits - m_qubit - 1\n"," control_qubit = [measurement_register[control_index]]\n"," for _ in range(2**m_qubit):\n"," qpe_circ.add_qcontrolbox(\n"," controlled_u_gate, control_qubit + list(state_prep_register)\n"," )\n","\n"," # Finally, append the inverse qft and measure the qubits\n"," qft_box = CircBox(build_qft_circuit(n_measurement_qubits))\n"," inverse_qft_box = qft_box.dagger\n"," qpe_circ.add_circbox(inverse_qft_box, list(measurement_register))\n"," qpe_circ.measure_register(measurement_register, \"c\")\n"," return qpe_circ"]},{"cell_type":"markdown","metadata":{},"source":["## Phase Estimation with a Trivial Eigenstate\n","\n","Lets test our circuit construction by preparing a trivial $|1\\rangle$ eigenstate of the $\\text{U1}$ gate. We can then see if our phase estimation circuit returns the expected eigenvalue."]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","U1(\\phi)|1\\rangle = e^{i\\phi} = e^{2 \\pi i \\theta} \\implies \\theta = \\frac{\\phi}{2}\n","\\end{equation}\n","$$\n","\n","So we expect that our ideal phase $\\theta$ will be half the input angle $\\phi$ to our $U1$ gate."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["prep_circuit = Circuit(1).X(0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["input_angle = 0.73 # angle as number of half turns"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["unitary_circuit = Circuit(1).U1(input_angle, 0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qpe_circ_trivial = build_phase_est_circuit(\n"," 4, state_prep_circuit=prep_circuit, unitary_circuit=unitary_circuit\n",")"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["render_circuit_jupyter(qpe_circ_trivial)"]},{"cell_type":"markdown","metadata":{},"source":["Lets use the noiseless `AerBackend` simulator to run our phase estimation circuit."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.extensions.qiskit import AerBackend"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["backend = AerBackend()"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["compiled_circ = backend.get_compiled_circuit(qpe_circ_trivial)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_shots = 1000\n","result = backend.run_circuit(compiled_circ, n_shots)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(result.get_counts())"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.backends.backendresult import BackendResult\n","import matplotlib.pyplot as plt"]},{"cell_type":"markdown","metadata":{},"source":["plotting function for QPE Notebook"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def plot_qpe_results(\n"," sim_result: BackendResult,\n"," n_strings: int = 4,\n"," dark_mode: bool = False,\n"," y_limit: int = 1000,\n",") -> None:\n"," \"\"\"\n"," Plots results in a barchart given a BackendResult. the number of stings displayed\n"," can be specified with the n_strings argument.\n"," \"\"\"\n"," counts_dict = sim_result.get_counts()\n"," sorted_shots = counts_dict.most_common()\n"," n_most_common_strings = sorted_shots[:n_strings]\n"," x_axis_values = [str(entry[0]) for entry in n_most_common_strings] # basis states\n"," y_axis_values = [entry[1] for entry in n_most_common_strings] # counts\n"," if dark_mode:\n"," plt.style.use(\"dark_background\")\n"," fig = plt.figure()\n"," ax = fig.add_axes((0, 0, 0.75, 0.5))\n"," color_list = [\"orange\"] * (len(x_axis_values))\n"," ax.bar(\n"," x=x_axis_values,\n"," height=y_axis_values,\n"," color=color_list,\n"," )\n"," ax.set_title(label=\"Results\")\n"," plt.ylim([0, y_limit])\n"," plt.xlabel(\"Basis State\")\n"," plt.ylabel(\"Number of Shots\")\n"," plt.show()"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plot_qpe_results(result, y_limit=int(1.2 * n_shots))"]},{"cell_type":"markdown","metadata":{},"source":["As expected we see one outcome with high probability. Lets now extract our approximation of $\\theta$ from our output bitstrings.\n","\n","suppose the $j$ is an integer representation of our most commonly measured bitstring."]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","\\theta_{estimate} = \\frac{j}{N}\n","\\end{equation}\n","$$"]},{"cell_type":"markdown","metadata":{},"source":["Here $N = 2 ^n$ where $n$ is the number of measurement qubits."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.backends.backendresult import BackendResult"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def single_phase_from_backendresult(result: BackendResult) -> float:\n"," # Extract most common measurement outcome\n"," basis_state = result.get_counts().most_common()[0][0]\n"," bitstring = \"\".join([str(bit) for bit in basis_state])\n"," integer = int(bitstring, 2)\n","\n"," # Calculate theta estimate\n"," return integer / (2 ** len(bitstring))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["theta = single_phase_from_backendresult(result)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(theta)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(input_angle / 2)"]},{"cell_type":"markdown","metadata":{},"source":["Our output is close to half our input angle $\\phi$ as expected. Lets calculate our error."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["error = round(abs(input_angle - (2 * theta)), 3)\n","print(error)"]},{"cell_type":"markdown","metadata":{},"source":["## Suggestions for further reading
\n","
\n","In this notebook we have shown the canonical variant of quantum phase estimation. There are several other variants.
\n","
\n","Quantinuum paper on Bayesian phase estimation -> https://arxiv.org/pdf/2306.16608.pdf
\n","Blog post on `ConjugationBox` -> https://tket.quantinuum.com/tket-blog/posts/controlled_gates/ - efficient circuits for controlled Pauli gadgets.
\n","
\n","As mentioned quantum phase estimation is a subroutine in Shor's algorithm. Read more about how phase estimation is used in period finding."]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} From 9cbf8f6c6f2f21dfcae6a815aa223a8e2e4aff77 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Tue, 7 Nov 2023 11:34:54 +0000 Subject: [PATCH 37/51] clean up pytket-qujax symbolic circuits --- examples/python/pytket-qujax_qaoa.py | 17 +++++++++++++++-- examples/pytket-qujax_qaoa.ipynb | 2 +- 2 files changed, 16 insertions(+), 3 deletions(-) diff --git a/examples/python/pytket-qujax_qaoa.py b/examples/python/pytket-qujax_qaoa.py index 05450f3b..980bef50 100644 --- a/examples/python/pytket-qujax_qaoa.py +++ b/examples/python/pytket-qujax_qaoa.py @@ -13,9 +13,16 @@ # # QAOA # The Quantum Approximate Optimization Algorithm (QAOA), first introduced by [Farhi et al.](https://arxiv.org/pdf/1411.4028.pdf), is a quantum variational algorithm used to solve optimization problems. It consists of a unitary $U(\beta, \gamma)$ formed by alternate repetitions of $U(\beta)=e^{-i\beta H_B}$ and $U(\gamma)=e^{-i\gamma H_P}$, where $H_B$ is the mixing Hamiltonian and $H_P$ the problem Hamiltonian. The goal is to find the optimal parameters that minimize $H_P$. # Given a depth $d$, the expression of the final unitary is $U(\beta, \gamma) = U(\beta_d)U(\gamma_d)\cdots U(\beta_1)U(\gamma_1)$. Notice that for each repetition the parameters are different. +# # ## Problem Hamiltonian # QAOA uses a problem dependent ansatz. Therefore, we first need to know the problem that we want to solve. In this case we will consider an Ising Hamiltonian with only $Z$ interactions. Given a set of pairs (or qubit indices) $E$, the problem Hamiltonian will be: -# $$H_P = \sum_{(i, j) \in E}\alpha_{ij}Z_iZ_j,$$ +# +# $$ +# \begin{equation} +# H_P = \sum_{(i, j) \in E}\alpha_{ij}Z_iZ_j, +# \end{equation} +# $$ +# # where $\alpha_{ij}$ are the coefficients. # Let's build our problem Hamiltonian with random coefficients and a set of pairs for a given number of qubits: @@ -36,7 +43,13 @@ # ## Variational Circuit # Before constructing the circuit, we still need to select the mixing Hamiltonian. In our case, we will be using $X$ gates in each qubit, so $H_B = \sum_{i=1}^{n}X_i$, where $n$ is the number of qubits. Notice that the unitary $U(\beta)$, given this mixing Hamiltonian, is an $X$ rotation in each qubit with angle $\beta$. # As for the unitary corresponding to the problem Hamiltonian, $U(\gamma)$, it has the following form: -# $$U(\gamma)=\prod_{(i, j) \in E}e^{-i\gamma\alpha_{ij}Z_iZ_j}$$ +# +# $$ +# \begin{equation} +# U(\gamma)=\prod_{(i, j) \in E}e^{-i\gamma\alpha_{ij}Z_i Z_j} +# \end{equation} +# $$ +# # The operation $e^{-i\gamma\alpha_{ij}Z_iZ_j}$ can be performed using two CNOT gates with qubit $i$ as control and qubit $j$ as target and a $Z$ rotation in qubit $j$ in between them, with angle $\gamma\alpha_{ij}$. # Finally, the initial state used, in general, with the QAOA is an equal superposition of all the basis states. This can be achieved adding a first layer of Hadamard gates in each qubit at the beginning of the circuit. diff --git a/examples/pytket-qujax_qaoa.ipynb b/examples/pytket-qujax_qaoa.ipynb index 066a6a12..a64e5812 100644 --- a/examples/pytket-qujax_qaoa.ipynb +++ b/examples/pytket-qujax_qaoa.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Symbolic circuits with `qujax` and `pytket-qujax`
\n", "In this notebook we will show how to manipulate symbolic circuits with the `pytket-qujax` extension. In particular, we will consider a QAOA and an Ising Hamiltonian."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit\n", "from pytket.circuit.display import render_circuit_jupyter\n", "from jax import numpy as jnp, random, value_and_grad, jit\n", "from sympy import Symbol\n", "import matplotlib.pyplot as plt"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import qujax\n", "from pytket.extensions.qujax import tk_to_qujax"]}, {"cell_type": "markdown", "metadata": {}, "source": ["# QAOA
\n", "The Quantum Approximate Optimization Algorithm (QAOA), first introduced by [Farhi et al.](https://arxiv.org/pdf/1411.4028.pdf), is a quantum variational algorithm used to solve optimization problems. It consists of a unitary $U(\\beta, \\gamma)$ formed by alternate repetitions of $U(\\beta)=e^{-i\\beta H_B}$ and $U(\\gamma)=e^{-i\\gamma H_P}$, where $H_B$ is the mixing Hamiltonian and $H_P$ the problem Hamiltonian. The goal is to find the optimal parameters that minimize $H_P$.
\n", "Given a depth $d$, the expression of the final unitary is $U(\\beta, \\gamma) = U(\\beta_d)U(\\gamma_d)\\cdots U(\\beta_1)U(\\gamma_1)$. Notice that for each repetition the parameters are different.
\n", "## Problem Hamiltonian
\n", "QAOA uses a problem dependent ansatz. Therefore, we first need to know the problem that we want to solve. In this case we will consider an Ising Hamiltonian with only $Z$ interactions. Given a set of pairs (or qubit indices) $E$, the problem Hamiltonian will be:
\n", "$$H_P = \\sum_{(i, j) \\in E}\\alpha_{ij}Z_iZ_j,$$
\n", "where $\\alpha_{ij}$ are the coefficients.
\n", "Let's build our problem Hamiltonian with random coefficients and a set of pairs for a given number of qubits:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["n_qubits = 4\n", "hamiltonian_qubit_inds = [(0, 1), (1, 2), (0, 2), (1, 3)]\n", "hamiltonian_gates = [[\"Z\", \"Z\"]] * (len(hamiltonian_qubit_inds))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Notice that in order to use the random package from jax we first need to define a seeded key"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["seed = 13\n", "key = random.PRNGKey(seed)\n", "coefficients = random.uniform(key, shape=(len(hamiltonian_qubit_inds),))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"Gates:\\t\", hamiltonian_gates)\n", "print(\"Qubits:\\t\", hamiltonian_qubit_inds)\n", "print(\"Coefficients:\\t\", coefficients)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Variational Circuit
\n", "Before constructing the circuit, we still need to select the mixing Hamiltonian. In our case, we will be using $X$ gates in each qubit, so $H_B = \\sum_{i=1}^{n}X_i$, where $n$ is the number of qubits. Notice that the unitary $U(\\beta)$, given this mixing Hamiltonian, is an $X$ rotation in each qubit with angle $\\beta$.
\n", "As for the unitary corresponding to the problem Hamiltonian, $U(\\gamma)$, it has the following form:
\n", "$$U(\\gamma)=\\prod_{(i, j) \\in E}e^{-i\\gamma\\alpha_{ij}Z_iZ_j}$$
\n", "The operation $e^{-i\\gamma\\alpha_{ij}Z_iZ_j}$ can be performed using two CNOT gates with qubit $i$ as control and qubit $j$ as target and a $Z$ rotation in qubit $j$ in between them, with angle $\\gamma\\alpha_{ij}$.
\n", "Finally, the initial state used, in general, with the QAOA is an equal superposition of all the basis states. This can be achieved adding a first layer of Hadamard gates in each qubit at the beginning of the circuit."]}, {"cell_type": "markdown", "metadata": {}, "source": ["With all the building blocks, let's construct the symbolic circuit using tket. Notice that in order to define the parameters, we use the ```Symbol``` object from the `sympy` package. More info can be found in this [documentation](https://cqcl.github.io/pytket/manual/manual_circuit.html#symbolic-circuits). In order to later convert the circuit to qujax, we need to return the list of symbolic parameters as well."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qaoa_circuit(n_qubits, depth):\n", " circuit = Circuit(n_qubits)\n", " p_keys = []\n\n", " # Initial State\n", " for i in range(n_qubits):\n", " circuit.H(i)\n", " for d in range(depth):\n", " # Hamiltonian unitary\n", " gamma_d = Symbol(f\"\u03b3_{d}\")\n", " for index in range(len(hamiltonian_qubit_inds)):\n", " pair = hamiltonian_qubit_inds[index]\n", " coef = coefficients[index]\n", " circuit.CX(pair[0], pair[1])\n", " circuit.Rz(gamma_d * coef, pair[1])\n", " circuit.CX(pair[0], pair[1])\n", " circuit.add_barrier(range(0, n_qubits))\n", " p_keys.append(gamma_d)\n\n", " # Mixing unitary\n", " beta_d = Symbol(f\"\u03b2_{d}\")\n", " for i in range(n_qubits):\n", " circuit.Rx(beta_d, i)\n", " p_keys.append(beta_d)\n", " return circuit, p_keys"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["depth = 3\n", "circuit, keys = qaoa_circuit(n_qubits, depth)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["keys"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Let's check the circuit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["render_circuit_jupyter(circuit)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["# Now for `qujax`
\n", "The `pytket.extensions.qujax.tk_to_qujax` function will generate a parameters -> statetensor function for us. However, in order to convert a symbolic circuit we first need to define the `symbol_map`. This object maps each symbol key to their corresponding index. In our case, since the object `keys` contains the symbols in the correct order, we can simply construct the dictionary as follows:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["symbol_map = {keys[i]: i for i in range(len(keys))}"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["symbol_map"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Then, we invoke the `tk_to_qujax` with both the circuit and the symbolic map."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["param_to_st = tk_to_qujax(circuit, symbol_map=symbol_map)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["And we also construct the expectation map using the problem Hamiltonian via qujax:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["st_to_expectation = qujax.get_statetensor_to_expectation_func(\n", " hamiltonian_gates, hamiltonian_qubit_inds, coefficients\n", ")"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["param_to_expectation = lambda param: st_to_expectation(param_to_st(param))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["# Training process
\n", "We construct a function that, given a parameter vector, returns the value of the cost function and the gradient.
\n", "We also `jit` to avoid recompilation, this means that the expensive `cost_and_grad` function is compiled once into a very fast XLA (C++) function which is then executed at each iteration. Alternatively, we could get the same speedup by replacing our `for` loop with `jax.lax.scan`. You can read more about JIT compilation in the [JAX documentation](https://jax.readthedocs.io/en/latest/jax-101/02-jitting.html)."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["cost_and_grad = jit(value_and_grad(param_to_expectation))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["For the training process we'll use vanilla gradient descent with a constant stepsize:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["seed = 123\n", "key = random.PRNGKey(seed)\n", "init_param = random.uniform(key, shape=(len(symbol_map),))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["n_steps = 150\n", "stepsize = 0.01"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["param = init_param"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["cost_vals = jnp.zeros(n_steps)\n", "cost_vals = cost_vals.at[0].set(param_to_expectation(init_param))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["for step in range(1, n_steps):\n", " cost_val, cost_grad = cost_and_grad(param)\n", " cost_vals = cost_vals.at[step].set(cost_val)\n", " param = param - stepsize * cost_grad\n", " print(\"Iteration:\", step, \"\\tCost:\", cost_val, end=\"\\r\")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Let's visualise the gradient descent"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["plt.plot(cost_vals)\n", "plt.xlabel(\"Iteration\")\n", "plt.ylabel(\"Cost\")"]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells":[{"cell_type":"markdown","metadata":{},"source":["# Symbolic circuits with `qujax` and `pytket-qujax`
\n","In this notebook we will show how to manipulate symbolic circuits with the `pytket-qujax` extension. In particular, we will consider a QAOA and an Ising Hamiltonian."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket import Circuit\n","from pytket.circuit.display import render_circuit_jupyter\n","from jax import numpy as jnp, random, value_and_grad, jit\n","from sympy import Symbol\n","import matplotlib.pyplot as plt"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["import qujax\n","from pytket.extensions.qujax import tk_to_qujax"]},{"cell_type":"markdown","metadata":{},"source":["# QAOA\n","The Quantum Approximate Optimization Algorithm (QAOA), first introduced by [Farhi et al.](https://arxiv.org/pdf/1411.4028.pdf), is a quantum variational algorithm used to solve optimization problems. It consists of a unitary $U(\\beta, \\gamma)$ formed by alternate repetitions of $U(\\beta)=e^{-i\\beta H_B}$ and $U(\\gamma)=e^{-i\\gamma H_P}$, where $H_B$ is the mixing Hamiltonian and $H_P$ the problem Hamiltonian. The goal is to find the optimal parameters that minimize $H_P$.\n","Given a depth $d$, the expression of the final unitary is $U(\\beta, \\gamma) = U(\\beta_d)U(\\gamma_d)\\cdots U(\\beta_1)U(\\gamma_1)$. Notice that for each repetition the parameters are different.\n","\n","## Problem Hamiltonian\n","QAOA uses a problem dependent ansatz. Therefore, we first need to know the problem that we want to solve. In this case we will consider an Ising Hamiltonian with only $Z$ interactions. Given a set of pairs (or qubit indices) $E$, the problem Hamiltonian will be:\n","\n","$$\n","\\begin{equation}\n","H_P = \\sum_{(i, j) \\in E}\\alpha_{ij}Z_iZ_j,\n","\\end{equation}\n","$$\n","\n","where $\\alpha_{ij}$ are the coefficients.\n","Let's build our problem Hamiltonian with random coefficients and a set of pairs for a given number of qubits:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_qubits = 4\n","hamiltonian_qubit_inds = [(0, 1), (1, 2), (0, 2), (1, 3)]\n","hamiltonian_gates = [[\"Z\", \"Z\"]] * (len(hamiltonian_qubit_inds))"]},{"cell_type":"markdown","metadata":{},"source":["Notice that in order to use the random package from jax we first need to define a seeded key"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["seed = 13\n","key = random.PRNGKey(seed)\n","coefficients = random.uniform(key, shape=(len(hamiltonian_qubit_inds),))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(\"Gates:\\t\", hamiltonian_gates)\n","print(\"Qubits:\\t\", hamiltonian_qubit_inds)\n","print(\"Coefficients:\\t\", coefficients)"]},{"cell_type":"markdown","metadata":{},"source":["## Variational Circuit\n","Before constructing the circuit, we still need to select the mixing Hamiltonian. In our case, we will be using $X$ gates in each qubit, so $H_B = \\sum_{i=1}^{n}X_i$, where $n$ is the number of qubits. Notice that the unitary $U(\\beta)$, given this mixing Hamiltonian, is an $X$ rotation in each qubit with angle $\\beta$.\n","As for the unitary corresponding to the problem Hamiltonian, $U(\\gamma)$, it has the following form:\n","\n","$$\n","\\begin{equation}\n","U(\\gamma)=\\prod_{(i, j) \\in E}e^{-i\\gamma\\alpha_{ij}Z_i Z_j}\n","\\end{equation}\n","$$\n","\n","The operation $e^{-i\\gamma\\alpha_{ij}Z_iZ_j}$ can be performed using two CNOT gates with qubit $i$ as control and qubit $j$ as target and a $Z$ rotation in qubit $j$ in between them, with angle $\\gamma\\alpha_{ij}$.\n","Finally, the initial state used, in general, with the QAOA is an equal superposition of all the basis states. This can be achieved adding a first layer of Hadamard gates in each qubit at the beginning of the circuit."]},{"cell_type":"markdown","metadata":{},"source":["With all the building blocks, let's construct the symbolic circuit using tket. Notice that in order to define the parameters, we use the ```Symbol``` object from the `sympy` package. More info can be found in this [documentation](https://cqcl.github.io/pytket/manual/manual_circuit.html#symbolic-circuits). In order to later convert the circuit to qujax, we need to return the list of symbolic parameters as well."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def qaoa_circuit(n_qubits, depth):\n"," circuit = Circuit(n_qubits)\n"," p_keys = []\n","\n"," # Initial State\n"," for i in range(n_qubits):\n"," circuit.H(i)\n"," for d in range(depth):\n"," # Hamiltonian unitary\n"," gamma_d = Symbol(f\"γ_{d}\")\n"," for index in range(len(hamiltonian_qubit_inds)):\n"," pair = hamiltonian_qubit_inds[index]\n"," coef = coefficients[index]\n"," circuit.CX(pair[0], pair[1])\n"," circuit.Rz(gamma_d * coef, pair[1])\n"," circuit.CX(pair[0], pair[1])\n"," circuit.add_barrier(range(0, n_qubits))\n"," p_keys.append(gamma_d)\n","\n"," # Mixing unitary\n"," beta_d = Symbol(f\"β_{d}\")\n"," for i in range(n_qubits):\n"," circuit.Rx(beta_d, i)\n"," p_keys.append(beta_d)\n"," return circuit, p_keys"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["depth = 3\n","circuit, keys = qaoa_circuit(n_qubits, depth)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["keys"]},{"cell_type":"markdown","metadata":{},"source":["Let's check the circuit:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["render_circuit_jupyter(circuit)"]},{"cell_type":"markdown","metadata":{},"source":["# Now for `qujax`
\n","The `pytket.extensions.qujax.tk_to_qujax` function will generate a parameters -> statetensor function for us. However, in order to convert a symbolic circuit we first need to define the `symbol_map`. This object maps each symbol key to their corresponding index. In our case, since the object `keys` contains the symbols in the correct order, we can simply construct the dictionary as follows:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["symbol_map = {keys[i]: i for i in range(len(keys))}"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["symbol_map"]},{"cell_type":"markdown","metadata":{},"source":["Then, we invoke the `tk_to_qujax` with both the circuit and the symbolic map."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_st = tk_to_qujax(circuit, symbol_map=symbol_map)"]},{"cell_type":"markdown","metadata":{},"source":["And we also construct the expectation map using the problem Hamiltonian via qujax:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["st_to_expectation = qujax.get_statetensor_to_expectation_func(\n"," hamiltonian_gates, hamiltonian_qubit_inds, coefficients\n",")"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_expectation = lambda param: st_to_expectation(param_to_st(param))"]},{"cell_type":"markdown","metadata":{},"source":["# Training process
\n","We construct a function that, given a parameter vector, returns the value of the cost function and the gradient.
\n","We also `jit` to avoid recompilation, this means that the expensive `cost_and_grad` function is compiled once into a very fast XLA (C++) function which is then executed at each iteration. Alternatively, we could get the same speedup by replacing our `for` loop with `jax.lax.scan`. You can read more about JIT compilation in the [JAX documentation](https://jax.readthedocs.io/en/latest/jax-101/02-jitting.html)."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["cost_and_grad = jit(value_and_grad(param_to_expectation))"]},{"cell_type":"markdown","metadata":{},"source":["For the training process we'll use vanilla gradient descent with a constant stepsize:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["seed = 123\n","key = random.PRNGKey(seed)\n","init_param = random.uniform(key, shape=(len(symbol_map),))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_steps = 150\n","stepsize = 0.01"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param = init_param"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["cost_vals = jnp.zeros(n_steps)\n","cost_vals = cost_vals.at[0].set(param_to_expectation(init_param))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["for step in range(1, n_steps):\n"," cost_val, cost_grad = cost_and_grad(param)\n"," cost_vals = cost_vals.at[step].set(cost_val)\n"," param = param - stepsize * cost_grad\n"," print(\"Iteration:\", step, \"\\tCost:\", cost_val, end=\"\\r\")"]},{"cell_type":"markdown","metadata":{},"source":["Let's visualise the gradient descent"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.plot(cost_vals)\n","plt.xlabel(\"Iteration\")\n","plt.ylabel(\"Cost\")"]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} From 4de735b37474033d8265e5257e478222fadfcd14 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Tue, 7 Nov 2023 12:21:25 +0000 Subject: [PATCH 38/51] finally fix dodgy math formatting in qujax classification example --- .../python/pytket-qujax-classification.py | 36 +++++++++++++++---- examples/pytket-qujax-classification.ipynb | 2 +- 2 files changed, 30 insertions(+), 8 deletions(-) diff --git a/examples/python/pytket-qujax-classification.py b/examples/python/pytket-qujax-classification.py index a8214a0c..0a9d279a 100644 --- a/examples/python/pytket-qujax-classification.py +++ b/examples/python/pytket-qujax-classification.py @@ -57,7 +57,13 @@ def classification_function(x, y): angles_to_st = tk_to_qujax(c) # We'll parameterise each angle as -# $$ \theta_k = b_k + w_k * x_k $$ +# +# $$ +# \begin{equation} +# \theta_k = b_k + w_k \, x_k +# \end{equation} +# $$ +# # where $b_k, w_k$ are variational parameters to be learnt and $x_k = x_0$ if $k$ even, $x_k = x_1$ if $k$ odd for a single bivariate input point $(x_0, x_1)$. n_angles = 3 * n_qubits * depth @@ -91,18 +97,34 @@ def param_and_x_to_probability(param, x_single): # For binary classification, the likelihood for our full data set $(x_{1:N}, y_{1:N})$ is -# $$ p(y_{1:N} \mid b, w, x_{1:N}) = \prod_{i=1}^N p(y_i \mid b, w, x_i) = \prod_{i=1}^N (1 - q_{(b,w)}(x_i))^{\mathbb{I}[y_i = 0]}q_{(b,w)}(x_i)^{\mathbb{I}[y_i = 1]}, $$ +# +# $$ +# \begin{equation} +# p(y_{1:N} \mid b, w, x_{1:N}) = \prod_{i=1}^N p(y_i \mid b, w, x_i) = \prod_{i=1}^N (1 - q_{(b,w)}(x_i))^{I[y_i = 0]}q_{(b,w)}(x_i)^{I[y_i = 1]}, +# \end{equation} +# $$ +# # where $q_{(b, w)}(x)$ is the probability the quantum circuit classifies input $x$ as donut given variational parameter vectors $(b, w)$. This gives log-likelihood -# $$ \log p(y_{1:N} \mid b, w, x_{1:N}) = \sum_{i=1}^N \mathbb{I}[y_i = 0] \log(1 - q_{(b,w)}(x_i)) + \mathbb{I}[y_i = 1] \log q_{(b,w)}(x_i), $$ +# +# $$ +# \begin{equation} +# \log p(y_{1:N} \mid b, w, x_{1:N}) = \sum_{i=1}^N I[y_i = 0] \log(1 - q_{(b,w)}(x_i)) + I[y_i = 1] \log q_{(b,w)}(x_i), +# \end{equation} +# $$ +# # which we would like to maximise. # # Unfortunately, the log-likelihood **cannot** be approximated unbiasedly using shots, that is we can approximate $q_{(b,w)}(x_i)$ unbiasedly but not $\log(q_{(b,w)}(x_i))$. # Note that in qujax simulations we can use the statetensor to calculate this exactly, but it is still good to keep in mind loss functions that can also be used with shots from a quantum device. - +# # Instead we can minimise an expected distance between shots and data -#
-# $$ C(b, w, x, y) = \mathbb{E}_{p(y' \mid q_{(b, w)}(x))}[\ell(y', y)] = (1 - q_{(b, w)}(x)) \ell(0, y) + q_{(b, w)}(x)\ell(1, y), $$ -#
+# +# $$ +# \begin{equation} +# C(b, w, x, y) = E_{p(y' \mid q_{(b, w)}(x))}[\ell(y', y)] = (1 - q_{(b, w)}(x)) \ell(0, y) + q_{(b, w)}(x)\ell(1, y), +# \end{equation} +# $$ +# # where $y'$ is a shot, $y$ is a data label and $\ell$ is some distance between bitstrings - here we simply set $\ell(0, 0) = \ell(1, 1) = 0$ and $\ell(0, 1) = \ell(1, 0) = 1$ (which coincides with the Hamming distance for this binary example). The full batch cost function is $C(b, w) = \frac1N \sum_{i=1}^N C(b, w, x_i, y_i)$. # # Note that to calculate the cost function we need to evaluate the statetensor for every input point $x_i$. If the dataset becomes too large, we can easily minibatch. diff --git a/examples/pytket-qujax-classification.ipynb b/examples/pytket-qujax-classification.ipynb index 115720dc..052e485d 100644 --- a/examples/pytket-qujax-classification.ipynb +++ b/examples/pytket-qujax-classification.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Binary classification using pytket-qujax"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from jax import numpy as jnp, random, vmap, value_and_grad, jit\n", "from pytket import Circuit\n", "from pytket.circuit.display import render_circuit_jupyter\n", "from pytket.extensions.qujax.qujax_convert import tk_to_qujax\n", "import matplotlib.pyplot as plt"]}, {"cell_type": "markdown", "metadata": {}, "source": ["# Define the classification task
\n", "We'll try and learn a _donut_ binary classification function (i.e. a bivariate coordinate is labelled 1 if it is inside the donut and 0 if it is outside)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["inner_rad = 0.25\n", "outer_rad = 0.75"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def classification_function(x, y):\n", " r = jnp.sqrt(x**2 + y**2)\n", " return jnp.where((r > inner_rad) * (r < outer_rad), 1, 0)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["linsp = jnp.linspace(-1, 1, 1000)\n", "Z = vmap(lambda x: vmap(lambda y: classification_function(x, y))(linsp))(linsp)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["plt.contourf(linsp, linsp, Z, cmap=\"Purples\")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now let's generate some data for our quantum circuit to learn from"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["n_data = 1000\n", "x = random.uniform(random.PRNGKey(0), shape=(n_data, 2), minval=-1, maxval=1)\n", "y = classification_function(x[:, 0], x[:, 1])"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["plt.scatter(x[:, 0], x[:, 1], alpha=jnp.where(y, 1, 0.2), s=10)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["# Quantum circuit time
\n", "We'll use a variant of data re-uploading [P\u00e9rez-Salinas et al](https://doi.org/10.22331/q-2020-02-06-226) to encode the input data, alongside some variational parameters within a quantum circuit classifier"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["n_qubits = 3\n", "depth = 5"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["c = Circuit(n_qubits)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["for layer in range(depth):\n", " for qi in range(n_qubits):\n", " c.Rz(0.0, qi)\n", " c.Ry(0.0, qi)\n", " c.Rz(0.0, qi)\n", " if layer < (depth - 1):\n", " for qi in range(layer, layer + n_qubits - 1, 2):\n", " c.CZ(qi % n_qubits, (qi + 1) % n_qubits)\n", " c.add_barrier(range(n_qubits))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["render_circuit_jupyter(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can use `pytket-qujax` to generate our angles-to-statetensor function."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["angles_to_st = tk_to_qujax(c)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We'll parameterise each angle as
\n", "$$ \\theta_k = b_k + w_k * x_k $$
\n", "where $b_k, w_k$ are variational parameters to be learnt and $x_k = x_0$ if $k$ even, $x_k = x_1$ if $k$ odd for a single bivariate input point $(x_0, x_1)$."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["n_angles = 3 * n_qubits * depth\n", "n_params = 2 * n_angles"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def param_and_x_to_angles(param, x_single):\n", " biases = param[:n_angles]\n", " weights = param[n_angles:]\n", " weights_times_data = jnp.where(\n", " jnp.arange(n_angles) % 2 == 0, weights * x_single[0], weights * x_single[1]\n", " )\n", " angles = biases + weights_times_data\n", " return angles"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["param_and_x_to_st = lambda param, x_single: angles_to_st(\n", " param_and_x_to_angles(param, x_single)\n", ")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We'll measure the first qubit only (if its 1 we label _donut_, if its 0 we label _not donut_)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def param_and_x_to_probability(param, x_single):\n", " st = param_and_x_to_st(param, x_single)\n", " all_probs = jnp.square(jnp.abs(st))\n", " first_qubit_probs = jnp.sum(all_probs, axis=range(1, n_qubits))\n", " return first_qubit_probs[1]"]}, {"cell_type": "markdown", "metadata": {}, "source": ["For binary classification, the likelihood for our full data set $(x_{1:N}, y_{1:N})$ is
\n", "$$ p(y_{1:N} \\mid b, w, x_{1:N}) = \\prod_{i=1}^N p(y_i \\mid b, w, x_i) = \\prod_{i=1}^N (1 - q_{(b,w)}(x_i))^{\\mathbb{I}[y_i = 0]}q_{(b,w)}(x_i)^{\\mathbb{I}[y_i = 1]}, $$
\n", "where $q_{(b, w)}(x)$ is the probability the quantum circuit classifies input $x$ as donut given variational parameter vectors $(b, w)$. This gives log-likelihood
\n", "$$ \\log p(y_{1:N} \\mid b, w, x_{1:N}) = \\sum_{i=1}^N \\mathbb{I}[y_i = 0] \\log(1 - q_{(b,w)}(x_i)) + \\mathbb{I}[y_i = 1] \\log q_{(b,w)}(x_i), $$
\n", "which we would like to maximise.
\n", "
\n", "Unfortunately, the log-likelihood **cannot** be approximated unbiasedly using shots, that is we can approximate $q_{(b,w)}(x_i)$ unbiasedly but not $\\log(q_{(b,w)}(x_i))$.
\n", "Note that in qujax simulations we can use the statetensor to calculate this exactly, but it is still good to keep in mind loss functions that can also be used with shots from a quantum device."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Instead we can minimise an expected distance between shots and data
\n", "

\n", "$$ C(b, w, x, y) = \\mathbb{E}_{p(y' \\mid q_{(b, w)}(x))}[\\ell(y', y)] = (1 - q_{(b, w)}(x)) \\ell(0, y) + q_{(b, w)}(x)\\ell(1, y), $$
\n", "

\n", "where $y'$ is a shot, $y$ is a data label and $\\ell$ is some distance between bitstrings - here we simply set $\\ell(0, 0) = \\ell(1, 1) = 0$ and $\\ell(0, 1) = \\ell(1, 0) = 1$ (which coincides with the Hamming distance for this binary example). The full batch cost function is $C(b, w) = \\frac1N \\sum_{i=1}^N C(b, w, x_i, y_i)$.
\n", "
\n", "Note that to calculate the cost function we need to evaluate the statetensor for every input point $x_i$. If the dataset becomes too large, we can easily minibatch."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def param_to_cost(param):\n", " donut_probs = vmap(param_and_x_to_probability, in_axes=(None, 0))(param, x)\n", " costs = jnp.where(y, 1 - donut_probs, donut_probs)\n", " return costs.mean()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["# Ready to descend some gradients?
\n", "We'll just use vanilla gradient descent here"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["param_to_cost_and_grad = jit(value_and_grad(param_to_cost))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["n_iter = 1000\n", "stepsize = 1e-1\n", "param = random.uniform(random.PRNGKey(1), shape=(n_params,), minval=0, maxval=2)\n", "costs = jnp.zeros(n_iter)\n", "for i in range(n_iter):\n", " cost, grad = param_to_cost_and_grad(param)\n", " costs = costs.at[i].set(cost)\n", " param = param - stepsize * grad\n", " print(i, \"Cost: \", cost, end=\"\\r\")"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["plt.plot(costs)\n", "plt.xlabel(\"Iteration\")\n", "plt.ylabel(\"Cost\")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["# Visualise trained classifier"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["linsp = jnp.linspace(-1, 1, 100)\n", "Z = vmap(\n", " lambda a: vmap(lambda b: param_and_x_to_probability(param, jnp.array([a, b])))(\n", " linsp\n", " )\n", ")(linsp)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["plt.contourf(linsp, linsp, Z, cmap=\"Purples\", alpha=0.8)\n", "circle_linsp = jnp.linspace(0, 2 * jnp.pi, 100)\n", "plt.plot(inner_rad * jnp.cos(circle_linsp), inner_rad * jnp.sin(circle_linsp), c=\"red\")\n", "plt.plot(outer_rad * jnp.cos(circle_linsp), outer_rad * jnp.sin(circle_linsp), c=\"red\")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Looks good, it has clearly grasped the donut shape. Sincerest apologies if you are now hungry! \ud83c\udf69"]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells":[{"cell_type":"markdown","metadata":{},"source":["# Binary classification using pytket-qujax"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from jax import numpy as jnp, random, vmap, value_and_grad, jit\n","from pytket import Circuit\n","from pytket.circuit.display import render_circuit_jupyter\n","from pytket.extensions.qujax.qujax_convert import tk_to_qujax\n","import matplotlib.pyplot as plt"]},{"cell_type":"markdown","metadata":{},"source":["# Define the classification task\n","We'll try and learn a _donut_ binary classification function (i.e. a bivariate coordinate is labelled 1 if it is inside the donut and 0 if it is outside)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["inner_rad = 0.25\n","outer_rad = 0.75"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def classification_function(x, y):\n"," r = jnp.sqrt(x**2 + y**2)\n"," return jnp.where((r > inner_rad) * (r < outer_rad), 1, 0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["linsp = jnp.linspace(-1, 1, 1000)\n","Z = vmap(lambda x: vmap(lambda y: classification_function(x, y))(linsp))(linsp)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.contourf(linsp, linsp, Z, cmap=\"Purples\")"]},{"cell_type":"markdown","metadata":{},"source":["Now let's generate some data for our quantum circuit to learn from"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_data = 1000\n","x = random.uniform(random.PRNGKey(0), shape=(n_data, 2), minval=-1, maxval=1)\n","y = classification_function(x[:, 0], x[:, 1])"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.scatter(x[:, 0], x[:, 1], alpha=jnp.where(y, 1, 0.2), s=10)"]},{"cell_type":"markdown","metadata":{},"source":["# Quantum circuit time\n","We'll use a variant of data re-uploading [Pérez-Salinas et al](https://doi.org/10.22331/q-2020-02-06-226) to encode the input data, alongside some variational parameters within a quantum circuit classifier"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_qubits = 3\n","depth = 5"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["c = Circuit(n_qubits)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["for layer in range(depth):\n"," for qi in range(n_qubits):\n"," c.Rz(0.0, qi)\n"," c.Ry(0.0, qi)\n"," c.Rz(0.0, qi)\n"," if layer < (depth - 1):\n"," for qi in range(layer, layer + n_qubits - 1, 2):\n"," c.CZ(qi % n_qubits, (qi + 1) % n_qubits)\n"," c.add_barrier(range(n_qubits))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["render_circuit_jupyter(c)"]},{"cell_type":"markdown","metadata":{},"source":["We can use `pytket-qujax` to generate our angles-to-statetensor function."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["angles_to_st = tk_to_qujax(c)"]},{"cell_type":"markdown","metadata":{},"source":["We'll parameterise each angle as\n","\n","$$\n","\\begin{equation}\n","\\theta_k = b_k + w_k \\, x_k\n","\\end{equation}\n","$$\n","\n","where $b_k, w_k$ are variational parameters to be learnt and $x_k = x_0$ if $k$ even, $x_k = x_1$ if $k$ odd for a single bivariate input point $(x_0, x_1)$."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_angles = 3 * n_qubits * depth\n","n_params = 2 * n_angles"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def param_and_x_to_angles(param, x_single):\n"," biases = param[:n_angles]\n"," weights = param[n_angles:]\n"," weights_times_data = jnp.where(\n"," jnp.arange(n_angles) % 2 == 0, weights * x_single[0], weights * x_single[1]\n"," )\n"," angles = biases + weights_times_data\n"," return angles"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_and_x_to_st = lambda param, x_single: angles_to_st(\n"," param_and_x_to_angles(param, x_single)\n",")"]},{"cell_type":"markdown","metadata":{},"source":["We'll measure the first qubit only (if its 1 we label _donut_, if its 0 we label _not donut_)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def param_and_x_to_probability(param, x_single):\n"," st = param_and_x_to_st(param, x_single)\n"," all_probs = jnp.square(jnp.abs(st))\n"," first_qubit_probs = jnp.sum(all_probs, axis=range(1, n_qubits))\n"," return first_qubit_probs[1]"]},{"cell_type":"markdown","metadata":{},"source":["For binary classification, the likelihood for our full data set $(x_{1:N}, y_{1:N})$ is\n","\n","$$\n","\\begin{equation}\n","p(y_{1:N} \\mid b, w, x_{1:N}) = \\prod_{i=1}^N p(y_i \\mid b, w, x_i) = \\prod_{i=1}^N (1 - q_{(b,w)}(x_i))^{I[y_i = 0]}q_{(b,w)}(x_i)^{I[y_i = 1]},\n","\\end{equation}\n","$$\n","\n","where $q_{(b, w)}(x)$ is the probability the quantum circuit classifies input $x$ as donut given variational parameter vectors $(b, w)$. This gives log-likelihood\n","\n","$$\n","\\begin{equation}\n"," \\log p(y_{1:N} \\mid b, w, x_{1:N}) = \\sum_{i=1}^N I[y_i = 0] \\log(1 - q_{(b,w)}(x_i)) + I[y_i = 1] \\log q_{(b,w)}(x_i),\n","\\end{equation}\n","$$\n","\n","which we would like to maximise.\n","\n","Unfortunately, the log-likelihood **cannot** be approximated unbiasedly using shots, that is we can approximate $q_{(b,w)}(x_i)$ unbiasedly but not $\\log(q_{(b,w)}(x_i))$.\n","Note that in qujax simulations we can use the statetensor to calculate this exactly, but it is still good to keep in mind loss functions that can also be used with shots from a quantum device.\n","\n","Instead we can minimise an expected distance between shots and data\n","\n","$$\n","\\begin{equation}\n","C(b, w, x, y) = E_{p(y' \\mid q_{(b, w)}(x))}[\\ell(y', y)] = (1 - q_{(b, w)}(x)) \\ell(0, y) + q_{(b, w)}(x)\\ell(1, y),\n","\\end{equation}\n","$$\n","\n","where $y'$ is a shot, $y$ is a data label and $\\ell$ is some distance between bitstrings - here we simply set $\\ell(0, 0) = \\ell(1, 1) = 0$ and $\\ell(0, 1) = \\ell(1, 0) = 1$ (which coincides with the Hamming distance for this binary example). The full batch cost function is $C(b, w) = \\frac1N \\sum_{i=1}^N C(b, w, x_i, y_i)$.\n","\n","Note that to calculate the cost function we need to evaluate the statetensor for every input point $x_i$. If the dataset becomes too large, we can easily minibatch."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def param_to_cost(param):\n"," donut_probs = vmap(param_and_x_to_probability, in_axes=(None, 0))(param, x)\n"," costs = jnp.where(y, 1 - donut_probs, donut_probs)\n"," return costs.mean()"]},{"cell_type":"markdown","metadata":{},"source":["# Ready to descend some gradients?\n","We'll just use vanilla gradient descent here"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_cost_and_grad = jit(value_and_grad(param_to_cost))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_iter = 1000\n","stepsize = 1e-1\n","param = random.uniform(random.PRNGKey(1), shape=(n_params,), minval=0, maxval=2)\n","costs = jnp.zeros(n_iter)\n","for i in range(n_iter):\n"," cost, grad = param_to_cost_and_grad(param)\n"," costs = costs.at[i].set(cost)\n"," param = param - stepsize * grad\n"," print(i, \"Cost: \", cost, end=\"\\r\")"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.plot(costs)\n","plt.xlabel(\"Iteration\")\n","plt.ylabel(\"Cost\")"]},{"cell_type":"markdown","metadata":{},"source":["# Visualise trained classifier"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["linsp = jnp.linspace(-1, 1, 100)\n","Z = vmap(\n"," lambda a: vmap(lambda b: param_and_x_to_probability(param, jnp.array([a, b])))(\n"," linsp\n"," )\n",")(linsp)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.contourf(linsp, linsp, Z, cmap=\"Purples\", alpha=0.8)\n","circle_linsp = jnp.linspace(0, 2 * jnp.pi, 100)\n","plt.plot(inner_rad * jnp.cos(circle_linsp), inner_rad * jnp.sin(circle_linsp), c=\"red\")\n","plt.plot(outer_rad * jnp.cos(circle_linsp), outer_rad * jnp.sin(circle_linsp), c=\"red\")"]},{"cell_type":"markdown","metadata":{},"source":["Looks good, it has clearly grasped the donut shape. Sincerest apologies if you are now hungry! 🍩"]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} From cf25d33d2d1c58b9cc533e217305983af9dbe87d Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Tue, 7 Nov 2023 12:23:24 +0000 Subject: [PATCH 39/51] add clean pytket-qujax heisenberg example --- examples/python/pytket-qujax_heisenberg_vqe.py | 8 +++++++- examples/pytket-qujax_heisenberg_vqe.ipynb | 2 +- 2 files changed, 8 insertions(+), 2 deletions(-) diff --git a/examples/python/pytket-qujax_heisenberg_vqe.py b/examples/python/pytket-qujax_heisenberg_vqe.py index fc840f49..2b367f1d 100644 --- a/examples/python/pytket-qujax_heisenberg_vqe.py +++ b/examples/python/pytket-qujax_heisenberg_vqe.py @@ -84,7 +84,13 @@ def get_circuit(n_qubits, depth): # - `coefficients`: A list of floats encoding any coefficients in the Hamiltonian. I.e. `[2.3, 0.8, 1.2]` corresponds to $a=2.3,b=0.8,c=1.2$ above. Must have the same length as the two above arguments. # More specifically let's consider the problem of finding the ground state of the quantum Heisenberg Hamiltonian -# $$ H = \sum_{i=1}^{n_\text{qubits}-1} X_i X_{i+1} + Y_i Y_{i+1} + Z_i Z_{i+1}. $$ + +# $$ +# \begin{equation} +# H = \sum_{i=1}^{n_\text{qubits}-1} X_i X_{i+1} + Y_i Y_{i+1} + Z_i Z_{i+1}. +# \end{equation} +# $$ +# # As described, we define the Hamiltonian via its gate strings, qubit indices and coefficients. hamiltonian_gates = [["X", "X"], ["Y", "Y"], ["Z", "Z"]] * (n_qubits - 1) diff --git a/examples/pytket-qujax_heisenberg_vqe.ipynb b/examples/pytket-qujax_heisenberg_vqe.ipynb index 7cfd274d..cc32ff66 100644 --- a/examples/pytket-qujax_heisenberg_vqe.ipynb +++ b/examples/pytket-qujax_heisenberg_vqe.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# VQE example with pytket-qujax"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from jax import numpy as jnp, random, value_and_grad, jit\n", "from pytket import Circuit\n", "from pytket.circuit.display import render_circuit_jupyter\n", "import matplotlib.pyplot as plt"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Let's start with a TKET circuit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import qujax\n", "from pytket.extensions.qujax.qujax_convert import tk_to_qujax"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We place barriers to stop tket automatically rearranging gates and we also store the number of circuit parameters as we'll need this later."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def get_circuit(n_qubits, depth):\n", " n_params = 2 * n_qubits * (depth + 1)\n", " param = jnp.zeros((n_params,))\n", " circuit = Circuit(n_qubits)\n", " k = 0\n", " for i in range(n_qubits):\n", " circuit.H(i)\n", " for i in range(n_qubits):\n", " circuit.Rx(param[k], i)\n", " k += 1\n", " for i in range(n_qubits):\n", " circuit.Ry(param[k], i)\n", " k += 1\n", " for _ in range(depth):\n", " for i in range(0, n_qubits - 1):\n", " circuit.CZ(i, i + 1)\n", " circuit.add_barrier(range(0, n_qubits))\n", " for i in range(n_qubits):\n", " circuit.Rx(param[k], i)\n", " k += 1\n", " for i in range(n_qubits):\n", " circuit.Ry(param[k], i)\n", " k += 1\n", " return circuit, n_params"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["n_qubits = 4\n", "depth = 2\n", "circuit, n_params = get_circuit(n_qubits, depth)\n", "render_circuit_jupyter(circuit)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Now let's invoke qujax
\n", "The `pytket.extensions.qujax.tk_to_qujax` function will generate a parameters -> statetensor function for us."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["param_to_st = tk_to_qujax(circuit)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Let's try it out on some random parameters values. Be aware that's JAX's random number generator requires a `jax.random.PRNGkey` every time it's called - more info on that [here](https://jax.readthedocs.io/en/latest/jax.random.html).
\n", "Be aware that we still have convention where parameters are specified as multiples of $\\pi$ - that is in [0,2]."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["params = random.uniform(random.PRNGKey(0), shape=(n_params,), minval=0.0, maxval=2.0)\n", "statetensor = param_to_st(params)\n", "print(statetensor)\n", "print(statetensor.shape)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Note that this function also has an optional second argument where an initiating `statetensor_in` can be provided. If it is not provided it will default to the all 0s state (as we use here)."]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can obtain statevector by simply calling `.flatten()`"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["statevector = statetensor.flatten()\n", "statevector.shape"]}, {"cell_type": "markdown", "metadata": {}, "source": ["And sampling probabilities by squaring the absolute value of the statevector"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["sample_probs = jnp.square(jnp.abs(statevector))\n", "plt.bar(jnp.arange(statevector.size), sample_probs)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Cost function"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now we have our `param_to_st` function we are free to define a cost function that acts on bitstrings (e.g. maxcut) or integers by directly wrapping a function around `param_to_st`. However, cost functions defined via quantum Hamiltonians are a bit more involved.
\n", "Fortunately, we can encode an Hamiltonian in JAX via the `qujax.get_statetensor_to_expectation_func` function which generates a statetensor -> expected value function for us.
\n", "It takes three arguments as input
\n", "- `gate_seq_seq`: A list of string (or array) lists encoding the gates in each term of the Hamiltonian. I.e. `[['X','X'], ['Y','Y'], ['Z','Z']]` corresponds to $H = aX_iX_j + bY_kY_l + cZ_mZ_n$ with qubit indices $i,j,k,l,m,n$ specified in the second argument and coefficients $a,b,c$ specified in the third argument
\n", "- `qubit_inds_seq`: A list of integer lists encoding which qubit indices to apply the aforementioned gates. I.e. `[[0, 1],[0,1],[0,1]]`. Must have the same structure as `gate_seq_seq` above.
\n", "- `coefficients`: A list of floats encoding any coefficients in the Hamiltonian. I.e. `[2.3, 0.8, 1.2]` corresponds to $a=2.3,b=0.8,c=1.2$ above. Must have the same length as the two above arguments."]}, {"cell_type": "markdown", "metadata": {}, "source": ["More specifically let's consider the problem of finding the ground state of the quantum Heisenberg Hamiltonian
\n", "$$ H = \\sum_{i=1}^{n_\\text{qubits}-1} X_i X_{i+1} + Y_i Y_{i+1} + Z_i Z_{i+1}. $$
\n", "As described, we define the Hamiltonian via its gate strings, qubit indices and coefficients."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian_gates = [[\"X\", \"X\"], [\"Y\", \"Y\"], [\"Z\", \"Z\"]] * (n_qubits - 1)\n", "hamiltonian_qubit_inds = [\n", " [int(i), int(i) + 1] for i in jnp.repeat(jnp.arange(n_qubits), 3)\n", "]\n", "coefficients = [1.0] * len(hamiltonian_qubit_inds)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"Gates:\\t\", hamiltonian_gates)\n", "print(\"Qubits:\\t\", hamiltonian_qubit_inds)\n", "print(\"Coefficients:\\t\", coefficients)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now let's get the Hamiltonian as a pure JAX function"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["st_to_expectation = qujax.get_statetensor_to_expectation_func(\n", " hamiltonian_gates, hamiltonian_qubit_inds, coefficients\n", ")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Let's check it works on the statetensor we've already generated."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["expected_val = st_to_expectation(statetensor)\n", "expected_val"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now let's wrap the `param_to_st` and `st_to_expectation` together to give us an all in one `param_to_expectation` cost function."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["param_to_expectation = lambda param: st_to_expectation(param_to_st(param))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["param_to_expectation(params)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Sanity check that a different, randomly generated set of parameters gives us a new expected value."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["new_params = random.uniform(\n", " random.PRNGKey(1), shape=(n_params,), minval=0.0, maxval=2.0\n", ")\n", "param_to_expectation(new_params)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Exact gradients within a VQE algorithm
\n", "The `param_to_expectation` function we created is a pure JAX function and outputs a scalar. This means we can pass it to `jax.grad` (or even better `jax.value_and_grad`)."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["cost_and_grad = value_and_grad(param_to_expectation)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The `cost_and_grad` function returns a tuple with the exact cost value and exact gradient evaluated at the parameters."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["cost_and_grad(params)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## Now we have all the tools we need to design our VQE!
\n", "We'll just use vanilla gradient descent with a constant stepsize"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def vqe(init_param, n_steps, stepsize):\n", " params = jnp.zeros((n_steps, n_params))\n", " params = params.at[0].set(init_param)\n", " cost_vals = jnp.zeros(n_steps)\n", " cost_vals = cost_vals.at[0].set(param_to_expectation(init_param))\n", " for step in range(1, n_steps):\n", " cost_val, cost_grad = cost_and_grad(params[step - 1])\n", " cost_vals = cost_vals.at[step].set(cost_val)\n", " new_param = params[step - 1] - stepsize * cost_grad\n", " params = params.at[step].set(new_param)\n", " print(\"Iteration:\", step, \"\\tCost:\", cost_val, end=\"\\r\")\n", " print(\"\\n\")\n", " return params, cost_vals"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Ok enough talking, let's run (and whilst we're at it we'll time it too)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["%time"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["vqe_params, vqe_cost_vals = vqe(params, n_steps=250, stepsize=0.01)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Let's plot the results..."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["plt.plot(vqe_cost_vals)\n", "plt.xlabel(\"Iteration\")\n", "plt.ylabel(\"Cost\")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Pretty good!"]}, {"cell_type": "markdown", "metadata": {}, "source": ["## `jax.jit` speedup
\n", "One last thing... We can significantly speed up the VQE above via the `jax.jit`. In our current implementation, the expensive `cost_and_grad` function is compiled to [XLA](https://www.tensorflow.org/xla) and then executed at each call. By invoking `jax.jit` we ensure that the function is compiled only once (on the first call) and then simply executed at each future call - this is much faster!"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["cost_and_grad = jit(cost_and_grad)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We'll demonstrate this using the second set of initial parameters we randomly generated (to be sure of no caching)."]}, {"cell_type": "markdown", "metadata": {}, "source": ["%time"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["new_vqe_params, new_vqe_cost_vals = vqe(new_params, n_steps=250, stepsize=0.01)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["That's some speedup!
\n", "But let's also plot the training to be sure it converged correctly"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["plt.plot(new_vqe_cost_vals)\n", "plt.xlabel(\"Iteration\")\n", "plt.ylabel(\"Cost\")"]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells":[{"cell_type":"markdown","metadata":{},"source":["# VQE example with pytket-qujax"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from jax import numpy as jnp, random, value_and_grad, jit\n","from pytket import Circuit\n","from pytket.circuit.display import render_circuit_jupyter\n","import matplotlib.pyplot as plt"]},{"cell_type":"markdown","metadata":{},"source":["## Let's start with a TKET circuit"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["import qujax\n","from pytket.extensions.qujax.qujax_convert import tk_to_qujax"]},{"cell_type":"markdown","metadata":{},"source":["We place barriers to stop tket automatically rearranging gates and we also store the number of circuit parameters as we'll need this later."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def get_circuit(n_qubits, depth):\n"," n_params = 2 * n_qubits * (depth + 1)\n"," param = jnp.zeros((n_params,))\n"," circuit = Circuit(n_qubits)\n"," k = 0\n"," for i in range(n_qubits):\n"," circuit.H(i)\n"," for i in range(n_qubits):\n"," circuit.Rx(param[k], i)\n"," k += 1\n"," for i in range(n_qubits):\n"," circuit.Ry(param[k], i)\n"," k += 1\n"," for _ in range(depth):\n"," for i in range(0, n_qubits - 1):\n"," circuit.CZ(i, i + 1)\n"," circuit.add_barrier(range(0, n_qubits))\n"," for i in range(n_qubits):\n"," circuit.Rx(param[k], i)\n"," k += 1\n"," for i in range(n_qubits):\n"," circuit.Ry(param[k], i)\n"," k += 1\n"," return circuit, n_params"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_qubits = 4\n","depth = 2\n","circuit, n_params = get_circuit(n_qubits, depth)\n","render_circuit_jupyter(circuit)"]},{"cell_type":"markdown","metadata":{},"source":["## Now let's invoke qujax\n","The `pytket.extensions.qujax.tk_to_qujax` function will generate a parameters -> statetensor function for us."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_st = tk_to_qujax(circuit)"]},{"cell_type":"markdown","metadata":{},"source":["Let's try it out on some random parameters values. Be aware that's JAX's random number generator requires a `jax.random.PRNGkey` every time it's called - more info on that [here](https://jax.readthedocs.io/en/latest/jax.random.html).\n","Be aware that we still have convention where parameters are specified as multiples of $\\pi$ - that is in [0,2]."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["params = random.uniform(random.PRNGKey(0), shape=(n_params,), minval=0.0, maxval=2.0)\n","statetensor = param_to_st(params)\n","print(statetensor)\n","print(statetensor.shape)"]},{"cell_type":"markdown","metadata":{},"source":["Note that this function also has an optional second argument where an initiating `statetensor_in` can be provided. If it is not provided it will default to the all 0s state (as we use here)."]},{"cell_type":"markdown","metadata":{},"source":["We can obtain statevector by simply calling `.flatten()`"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["statevector = statetensor.flatten()\n","statevector.shape"]},{"cell_type":"markdown","metadata":{},"source":["And sampling probabilities by squaring the absolute value of the statevector"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["sample_probs = jnp.square(jnp.abs(statevector))\n","plt.bar(jnp.arange(statevector.size), sample_probs)"]},{"cell_type":"markdown","metadata":{},"source":["## Cost function"]},{"cell_type":"markdown","metadata":{},"source":["Now we have our `param_to_st` function we are free to define a cost function that acts on bitstrings (e.g. maxcut) or integers by directly wrapping a function around `param_to_st`. However, cost functions defined via quantum Hamiltonians are a bit more involved.\n","Fortunately, we can encode an Hamiltonian in JAX via the `qujax.get_statetensor_to_expectation_func` function which generates a statetensor -> expected value function for us.\n","It takes three arguments as input\n","- `gate_seq_seq`: A list of string (or array) lists encoding the gates in each term of the Hamiltonian. I.e. `[['X','X'], ['Y','Y'], ['Z','Z']]` corresponds to $H = aX_iX_j + bY_kY_l + cZ_mZ_n$ with qubit indices $i,j,k,l,m,n$ specified in the second argument and coefficients $a,b,c$ specified in the third argument\n","- `qubit_inds_seq`: A list of integer lists encoding which qubit indices to apply the aforementioned gates. I.e. `[[0, 1],[0,1],[0,1]]`. Must have the same structure as `gate_seq_seq` above.\n","- `coefficients`: A list of floats encoding any coefficients in the Hamiltonian. I.e. `[2.3, 0.8, 1.2]` corresponds to $a=2.3,b=0.8,c=1.2$ above. Must have the same length as the two above arguments."]},{"cell_type":"markdown","metadata":{},"source":["More specifically let's consider the problem of finding the ground state of the quantum Heisenberg Hamiltonian"]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","H = \\sum_{i=1}^{n_\\text{qubits}-1} X_i X_{i+1} + Y_i Y_{i+1} + Z_i Z_{i+1}.\n","\\end{equation}\n","$$\n","\n","As described, we define the Hamiltonian via its gate strings, qubit indices and coefficients."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["hamiltonian_gates = [[\"X\", \"X\"], [\"Y\", \"Y\"], [\"Z\", \"Z\"]] * (n_qubits - 1)\n","hamiltonian_qubit_inds = [\n"," [int(i), int(i) + 1] for i in jnp.repeat(jnp.arange(n_qubits), 3)\n","]\n","coefficients = [1.0] * len(hamiltonian_qubit_inds)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(\"Gates:\\t\", hamiltonian_gates)\n","print(\"Qubits:\\t\", hamiltonian_qubit_inds)\n","print(\"Coefficients:\\t\", coefficients)"]},{"cell_type":"markdown","metadata":{},"source":["Now let's get the Hamiltonian as a pure JAX function"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["st_to_expectation = qujax.get_statetensor_to_expectation_func(\n"," hamiltonian_gates, hamiltonian_qubit_inds, coefficients\n",")"]},{"cell_type":"markdown","metadata":{},"source":["Let's check it works on the statetensor we've already generated."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["expected_val = st_to_expectation(statetensor)\n","expected_val"]},{"cell_type":"markdown","metadata":{},"source":["Now let's wrap the `param_to_st` and `st_to_expectation` together to give us an all in one `param_to_expectation` cost function."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_expectation = lambda param: st_to_expectation(param_to_st(param))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_expectation(params)"]},{"cell_type":"markdown","metadata":{},"source":["Sanity check that a different, randomly generated set of parameters gives us a new expected value."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["new_params = random.uniform(\n"," random.PRNGKey(1), shape=(n_params,), minval=0.0, maxval=2.0\n",")\n","param_to_expectation(new_params)"]},{"cell_type":"markdown","metadata":{},"source":["## Exact gradients within a VQE algorithm\n","The `param_to_expectation` function we created is a pure JAX function and outputs a scalar. This means we can pass it to `jax.grad` (or even better `jax.value_and_grad`)."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["cost_and_grad = value_and_grad(param_to_expectation)"]},{"cell_type":"markdown","metadata":{},"source":["The `cost_and_grad` function returns a tuple with the exact cost value and exact gradient evaluated at the parameters."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["cost_and_grad(params)"]},{"cell_type":"markdown","metadata":{},"source":["## Now we have all the tools we need to design our VQE!\n","We'll just use vanilla gradient descent with a constant stepsize"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def vqe(init_param, n_steps, stepsize):\n"," params = jnp.zeros((n_steps, n_params))\n"," params = params.at[0].set(init_param)\n"," cost_vals = jnp.zeros(n_steps)\n"," cost_vals = cost_vals.at[0].set(param_to_expectation(init_param))\n"," for step in range(1, n_steps):\n"," cost_val, cost_grad = cost_and_grad(params[step - 1])\n"," cost_vals = cost_vals.at[step].set(cost_val)\n"," new_param = params[step - 1] - stepsize * cost_grad\n"," params = params.at[step].set(new_param)\n"," print(\"Iteration:\", step, \"\\tCost:\", cost_val, end=\"\\r\")\n"," print(\"\\n\")\n"," return params, cost_vals"]},{"cell_type":"markdown","metadata":{},"source":["Ok enough talking, let's run (and whilst we're at it we'll time it too)"]},{"cell_type":"markdown","metadata":{},"source":["%time"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["vqe_params, vqe_cost_vals = vqe(params, n_steps=250, stepsize=0.01)"]},{"cell_type":"markdown","metadata":{},"source":["Let's plot the results..."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.plot(vqe_cost_vals)\n","plt.xlabel(\"Iteration\")\n","plt.ylabel(\"Cost\")"]},{"cell_type":"markdown","metadata":{},"source":["Pretty good!"]},{"cell_type":"markdown","metadata":{},"source":["## `jax.jit` speedup\n","One last thing... We can significantly speed up the VQE above via the `jax.jit`. In our current implementation, the expensive `cost_and_grad` function is compiled to [XLA](https://www.tensorflow.org/xla) and then executed at each call. By invoking `jax.jit` we ensure that the function is compiled only once (on the first call) and then simply executed at each future call - this is much faster!"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["cost_and_grad = jit(cost_and_grad)"]},{"cell_type":"markdown","metadata":{},"source":["We'll demonstrate this using the second set of initial parameters we randomly generated (to be sure of no caching)."]},{"cell_type":"markdown","metadata":{},"source":["%time"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["new_vqe_params, new_vqe_cost_vals = vqe(new_params, n_steps=250, stepsize=0.01)"]},{"cell_type":"markdown","metadata":{},"source":["That's some speedup!\n","But let's also plot the training to be sure it converged correctly"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.plot(new_vqe_cost_vals)\n","plt.xlabel(\"Iteration\")\n","plt.ylabel(\"Cost\")"]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} From 5722842926181cca49ea3a36d4be3ec463b5e0b6 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Tue, 7 Nov 2023 12:28:53 +0000 Subject: [PATCH 40/51] add clean notebooks --- examples/phase_estimation.ipynb | 2 +- examples/python/pytket-qujax-classification.py | 10 +++++++++- examples/pytket-qujax-classification.ipynb | 2 +- examples/pytket-qujax_qaoa.ipynb | 2 +- 4 files changed, 12 insertions(+), 4 deletions(-) diff --git a/examples/phase_estimation.ipynb b/examples/phase_estimation.ipynb index a3c2455d..f68b02f8 100644 --- a/examples/phase_estimation.ipynb +++ b/examples/phase_estimation.ipynb @@ -1 +1 @@ -{"cells":[{"cell_type":"markdown","metadata":{},"source":["# Quantum Phase Estimation\n","\n","When constructing circuits for quantum algorithms it is useful to think of higher level operations than just individual quantum gates.\n","\n","In `pytket` we can construct circuits using box structures which abstract away the complexity of the underlying circuit.\n","\n","This notebook is intended to complement the [boxes section](https://tket.quantinuum.com/user-manual/manual_circuit.html#boxes) of the user manual which introduces the different box types.\n","\n","To demonstrate boxes in `pytket` we will consider the Quantum Phase Estimation algorithm (QPE). This is an important subroutine in several quantum algorithms including Shor's algorithm and fault-tolerant approaches to quantum chemistry.\n","\n","## Overview of Phase Estimation\n","\n","The Quantum Phase Estimation algorithm can be used to estimate the eigenvalues of some unitary operator $U$ to some desired precision.\n","\n","The eigenvalues of $U$ lie on the unit circle, giving us the following eigenvalue equation\n","\n","$$\n","\\begin{equation}\n","U |\\psi \\rangle = e^{2 \\pi i \\theta} |\\psi\\rangle\\,, \\quad 0 \\leq \\theta \\leq 1\n","\\end{equation}\n","$$\n","\n","Here $|\\psi \\rangle$ is an eigenstate of the operator $U$. In phase estimation we estimate the eigenvalue $e^{2 \\pi i \\theta}$ by approximating $\\theta$.\n","\n","\n","The circuit for Quantum phase estimation is itself composed of several subroutines which we can realise as boxes.\n","\n","![](images/phase_est.png \"Quantum Phase Estimation Circuit\")"]},{"cell_type":"markdown","metadata":{},"source":["QPE is generally split up into three stages
\n","
\n","1. Firstly we prepare an initial state in one register. In parallel we prepare a uniform superposition state using Hadamard gates on some ancilla qubits. The number of ancilla qubits determines how precisely we can estimate the phase $\\theta$.
\n","
\n","2. Secondly we apply successive controlled $U$ gates. This has the effect of \"kicking back\" phases onto the ancilla qubits according to the eigenvalue equation above.
\n","
\n","3. Finally we apply the inverse Quantum Fourier Transform (QFT). This essentially plays the role of destructive interference, suppressing amplitudes from \"undesirable states\" and hopefully allowing us to measure a single outcome (or a small number of outcomes) with high probability.
\n","
\n","
\n","There is some subtlety around the first point. The initial state used can be an exact eigenstate of $U$ however this may be difficult to prepare if we don't know the eigenvalues of $U$ in advance. Alternatively we could use an initial state that is a linear combination of eigenstates, as the phase estimation will project into the eigenspace of $U$."]},{"cell_type":"markdown","metadata":{},"source":["We also assume that we can implement $U$ with a quantum circuit. In chemistry applications $U$ could be of the form $U=e^{-iHt}$ where $H$ is the Hamiltonian of some system of interest. In the cannonical algorithm, the number of controlled unitaries we apply scales exponentially with the number of ancilla qubits. This allows more precision at the expense of a larger quantum circuit."]},{"cell_type":"markdown","metadata":{},"source":["## The Quantum Fourier Transform"]},{"cell_type":"markdown","metadata":{},"source":["Before considering the other parts of the QPE algorithm, lets focus on the Quantum Fourier Transform (QFT) subroutine.\n","\n","Mathematically, the QFT has the following action.\n","\n","$$\n","\\begin{equation}\n","QFT : |j\\rangle\\ \\longmapsto \\sum_{k=0}^{N - 1} e^{2 \\pi ijk/N}|k\\rangle, \\quad N= 2^k\n","\\end{equation}\n","$$\n","\n","This is essentially the Discrete Fourier transform except the input is a quantum state $|j\\rangle$.\n","\n","It is well known that the QFT can be implemented efficiently with a quantum circuit\n","\n","We can build the circuit for the $n$ qubit QFT using $n$ Hadamard gates $\\frac{n}{2}$ swap gates and $\\frac{n(n-1)}{2}$ controlled unitary rotations $\\text{CU1}$.\n","\n","$$\n"," \\begin{equation}\n"," CU1(\\phi) =\n"," \\begin{pmatrix}\n"," I & 0 \\\\\n"," 0 & U1(\\phi)\n"," \\end{pmatrix}\n"," \\,, \\quad\n","U1(\\phi) =\n"," \\begin{pmatrix}\n"," 1 & 0 \\\\\n"," 0 & e^{i \\phi}\n"," \\end{pmatrix}\n"," \\end{equation}\n","$$\n","\n","The circuit for the Quantum Fourier transform on three qubits is the following\n","\n","![](images/qft.png \"QFT Circuit\")\n","\n","We can build this circuit in `pytket` by adding gate operations manually:"]},{"cell_type":"markdown","metadata":{},"source":["lets build the QFT for three qubits"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import Circuit\n","from pytket.circuit.display import render_circuit_jupyter"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft3_circ = Circuit(3)\n","qft3_circ.H(0)\n","qft3_circ.CU1(0.5, 1, 0)\n","qft3_circ.CU1(0.25, 2, 0)\n","qft3_circ.H(1)\n","qft3_circ.CU1(0.5, 2, 1)\n","qft3_circ.H(2)\n","qft3_circ.SWAP(0, 2)\n","render_circuit_jupyter(qft3_circ)"]},{"cell_type":"markdown","metadata":{},"source":["We can generalise the quantum Fourier transform to $n$ qubits by iterating over the qubits as follows"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def build_qft_circuit(n_qubits: int) -> Circuit:\n"," circ = Circuit(n_qubits, name=\"QFT\")\n"," for i in range(n_qubits):\n"," circ.H(i)\n"," for j in range(i + 1, n_qubits):\n"," circ.CU1(1 / 2 ** (j - i), j, i)\n"," for k in range(0, n_qubits // 2):\n"," circ.SWAP(k, n_qubits - k - 1)\n"," return circ"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft4_circ: Circuit = build_qft_circuit(4)\n","render_circuit_jupyter(qft4_circ)"]},{"cell_type":"markdown","metadata":{},"source":["Now that we have the generalised circuit we can wrap it up in a `CircBox` which can then be added to another circuit as a subroutine."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import CircBox"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft4_box: CircBox = CircBox(qft4_circ)\n","qft_circ = Circuit(4).add_gate(qft4_box, [0, 1, 2, 3])\n","render_circuit_jupyter(qft_circ)"]},{"cell_type":"markdown","metadata":{},"source":["Note how the `CircBox` inherits the name `QFT` from the underlying circuit."]},{"cell_type":"markdown","metadata":{},"source":["Recall that in our phase estimation algorithm we need to use the inverse QFT.\n","\n","$$\n","\\begin{equation}\n","\\text{QFT}^† : \\sum_{k=0}^{N - 1} e^{2 \\pi ijk/N}|k\\rangle \\longmapsto |j\\rangle\\,, \\quad N= 2^k\n","\\end{equation}\n","$$\n","\n","\n","Now that we have the QFT circuit we can obtain the inverse by using `CircBox.dagger`. We can also verify that this is correct by inspecting the circuit inside with `CircBox.get_circuit()`."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["inv_qft4_box = qft4_box.dagger\n","render_circuit_jupyter(inv_qft4_box.get_circuit())"]},{"cell_type":"markdown","metadata":{},"source":["## The Controlled Unitary Operations"]},{"cell_type":"markdown","metadata":{},"source":["In the phase estimation algorithm we repeatedly perform controlled unitary operations. In the canonical variant, the number of controlled unitaries will be $2^m - 1$ where $m$ is the number of measurement qubits."]},{"cell_type":"markdown","metadata":{},"source":["The form of $U$ will vary depending on the application. For chemistry or condensed matter physics $U$ typically be the time evolution operator $U(t) = e^{- i H t}$ where $H$ is the problem Hamiltonian."]},{"cell_type":"markdown","metadata":{},"source":["Suppose that we had the following decomposition for $H$ in terms of Pauli strings $P_j$ and complex coefficients $\\alpha_j$.\n","\n","$$\n","\\begin{equation}\n","H = \\sum_j \\alpha_j P_j\\,, \\quad \\, P_j \\in \\{I, \\,X, \\,Y, \\,Z\\}^{\\otimes n}\n","\\end{equation}\n","$$\n","\n","Here Pauli strings refers to tensor products of Pauli operators. These strings form an orthonormal basis for $2^n \\times 2^n$ matrices."]},{"cell_type":"markdown","metadata":{},"source":["If we have a Hamiltonian in the form above, we can then implement $U(t)$ as a sequence of Pauli gadget circuits. We can do this with the [PauliExpBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.PauliExpBox) construct in pytket. For more on `PauliExpBox` see the [user manual](https://tket.quantinuum.com/user-manual/manual_circuit.html#pauli-exponential-boxes)."]},{"cell_type":"markdown","metadata":{},"source":["Once we have a circuit to implement our time evolution operator $U(t)$, we can construct the controlled $U(t)$ operations using [QControlBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.QControlBox). If our base unitary is a sequence of `PauliExpBox`(es) then there is some structure we can exploit to simplify our circuit. See this [blog post](https://tket.quantinuum.com/tket-blog/posts/controlled_gates/) on [ConjugationBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.ConjugationBox) for more."]},{"cell_type":"markdown","metadata":{},"source":["In what follows, we will just construct a simplified instance of QPE where the controlled unitaries are just $\\text{CU1}$ gates."]},{"cell_type":"markdown","metadata":{},"source":["## Putting it all together"]},{"cell_type":"markdown","metadata":{},"source":["We can now define a function to build our entire QPE circuit. We can make this function take a state preparation circuit and a unitary circuit as input as well. The function also has the number of measurement qubits as input which will determine the precision of our phase estimate."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import QControlBox"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def build_phase_est_circuit(\n"," n_measurement_qubits: int, state_prep_circuit: Circuit, unitary_circuit: Circuit\n",") -> Circuit:\n"," qpe_circ: Circuit = Circuit()\n"," n_state_prep_qubits = state_prep_circuit.n_qubits\n"," measurement_register = qpe_circ.add_q_register(\"m\", n_measurement_qubits)\n"," state_prep_register = qpe_circ.add_q_register(\"p\", n_state_prep_qubits)\n"," qpe_circ.add_circuit(state_prep_circuit, list(state_prep_register))\n","\n"," # Create a controlled unitary with a single control qubit\n"," unitary_circuit.name = \"U\"\n"," controlled_u_gate = QControlBox(CircBox(unitary_circuit), 1)\n","\n"," # Add Hadamard gates to every qubit in the measurement register\n"," for m_qubit in measurement_register:\n"," qpe_circ.H(m_qubit)\n","\n"," # Add all (2**n_measurement_qubits - 1) of the controlled unitaries sequentially\n"," for m_qubit in range(n_measurement_qubits):\n"," control_index = n_measurement_qubits - m_qubit - 1\n"," control_qubit = [measurement_register[control_index]]\n"," for _ in range(2**m_qubit):\n"," qpe_circ.add_qcontrolbox(\n"," controlled_u_gate, control_qubit + list(state_prep_register)\n"," )\n","\n"," # Finally, append the inverse qft and measure the qubits\n"," qft_box = CircBox(build_qft_circuit(n_measurement_qubits))\n"," inverse_qft_box = qft_box.dagger\n"," qpe_circ.add_circbox(inverse_qft_box, list(measurement_register))\n"," qpe_circ.measure_register(measurement_register, \"c\")\n"," return qpe_circ"]},{"cell_type":"markdown","metadata":{},"source":["## Phase Estimation with a Trivial Eigenstate\n","\n","Lets test our circuit construction by preparing a trivial $|1\\rangle$ eigenstate of the $\\text{U1}$ gate. We can then see if our phase estimation circuit returns the expected eigenvalue."]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","U1(\\phi)|1\\rangle = e^{i\\phi} = e^{2 \\pi i \\theta} \\implies \\theta = \\frac{\\phi}{2}\n","\\end{equation}\n","$$\n","\n","So we expect that our ideal phase $\\theta$ will be half the input angle $\\phi$ to our $U1$ gate."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["prep_circuit = Circuit(1).X(0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["input_angle = 0.73 # angle as number of half turns"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["unitary_circuit = Circuit(1).U1(input_angle, 0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qpe_circ_trivial = build_phase_est_circuit(\n"," 4, state_prep_circuit=prep_circuit, unitary_circuit=unitary_circuit\n",")"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["render_circuit_jupyter(qpe_circ_trivial)"]},{"cell_type":"markdown","metadata":{},"source":["Lets use the noiseless `AerBackend` simulator to run our phase estimation circuit."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.extensions.qiskit import AerBackend"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["backend = AerBackend()"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["compiled_circ = backend.get_compiled_circuit(qpe_circ_trivial)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_shots = 1000\n","result = backend.run_circuit(compiled_circ, n_shots)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(result.get_counts())"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.backends.backendresult import BackendResult\n","import matplotlib.pyplot as plt"]},{"cell_type":"markdown","metadata":{},"source":["plotting function for QPE Notebook"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def plot_qpe_results(\n"," sim_result: BackendResult,\n"," n_strings: int = 4,\n"," dark_mode: bool = False,\n"," y_limit: int = 1000,\n",") -> None:\n"," \"\"\"\n"," Plots results in a barchart given a BackendResult. the number of stings displayed\n"," can be specified with the n_strings argument.\n"," \"\"\"\n"," counts_dict = sim_result.get_counts()\n"," sorted_shots = counts_dict.most_common()\n"," n_most_common_strings = sorted_shots[:n_strings]\n"," x_axis_values = [str(entry[0]) for entry in n_most_common_strings] # basis states\n"," y_axis_values = [entry[1] for entry in n_most_common_strings] # counts\n"," if dark_mode:\n"," plt.style.use(\"dark_background\")\n"," fig = plt.figure()\n"," ax = fig.add_axes((0, 0, 0.75, 0.5))\n"," color_list = [\"orange\"] * (len(x_axis_values))\n"," ax.bar(\n"," x=x_axis_values,\n"," height=y_axis_values,\n"," color=color_list,\n"," )\n"," ax.set_title(label=\"Results\")\n"," plt.ylim([0, y_limit])\n"," plt.xlabel(\"Basis State\")\n"," plt.ylabel(\"Number of Shots\")\n"," plt.show()"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plot_qpe_results(result, y_limit=int(1.2 * n_shots))"]},{"cell_type":"markdown","metadata":{},"source":["As expected we see one outcome with high probability. Lets now extract our approximation of $\\theta$ from our output bitstrings.\n","\n","suppose the $j$ is an integer representation of our most commonly measured bitstring."]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","\\theta_{estimate} = \\frac{j}{N}\n","\\end{equation}\n","$$"]},{"cell_type":"markdown","metadata":{},"source":["Here $N = 2 ^n$ where $n$ is the number of measurement qubits."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.backends.backendresult import BackendResult"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def single_phase_from_backendresult(result: BackendResult) -> float:\n"," # Extract most common measurement outcome\n"," basis_state = result.get_counts().most_common()[0][0]\n"," bitstring = \"\".join([str(bit) for bit in basis_state])\n"," integer = int(bitstring, 2)\n","\n"," # Calculate theta estimate\n"," return integer / (2 ** len(bitstring))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["theta = single_phase_from_backendresult(result)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(theta)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(input_angle / 2)"]},{"cell_type":"markdown","metadata":{},"source":["Our output is close to half our input angle $\\phi$ as expected. Lets calculate our error."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["error = round(abs(input_angle - (2 * theta)), 3)\n","print(error)"]},{"cell_type":"markdown","metadata":{},"source":["## Suggestions for further reading
\n","
\n","In this notebook we have shown the canonical variant of quantum phase estimation. There are several other variants.
\n","
\n","Quantinuum paper on Bayesian phase estimation -> https://arxiv.org/pdf/2306.16608.pdf
\n","Blog post on `ConjugationBox` -> https://tket.quantinuum.com/tket-blog/posts/controlled_gates/ - efficient circuits for controlled Pauli gadgets.
\n","
\n","As mentioned quantum phase estimation is a subroutine in Shor's algorithm. Read more about how phase estimation is used in period finding."]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} +{"cells":[{"cell_type":"markdown","metadata":{},"source":["# Quantum Phase Estimation\n","\n","When constructing circuits for quantum algorithms it is useful to think of higher level operations than just individual quantum gates.\n","\n","In `pytket` we can construct circuits using box structures which abstract away the complexity of the underlying circuit.\n","\n","This notebook is intended to complement the [boxes section](https://tket.quantinuum.com/user-manual/manual_circuit.html#boxes) of the user manual which introduces the different box types.\n","\n","To demonstrate boxes in `pytket` we will consider the Quantum Phase Estimation algorithm (QPE). This is an important subroutine in several quantum algorithms including Shor's algorithm and fault-tolerant approaches to quantum chemistry.\n","\n","## Overview of Phase Estimation\n","\n","The Quantum Phase Estimation algorithm can be used to estimate the eigenvalues of some unitary operator $U$ to some desired precision.\n","\n","The eigenvalues of $U$ lie on the unit circle, giving us the following eigenvalue equation\n","\n","$$\n","\\begin{equation}\n","U |\\psi \\rangle = e^{2 \\pi i \\theta} |\\psi\\rangle\\,, \\quad 0 \\leq \\theta \\leq 1\n","\\end{equation}\n","$$\n","\n","Here $|\\psi \\rangle$ is an eigenstate of the operator $U$. In phase estimation we estimate the eigenvalue $e^{2 \\pi i \\theta}$ by approximating $\\theta$.\n","\n","\n","The circuit for Quantum phase estimation is itself composed of several subroutines which we can realise as boxes.\n","\n","![](images/phase_est.png \"Quantum Phase Estimation Circuit\")"]},{"cell_type":"markdown","metadata":{},"source":["QPE is generally split up into three stages\n","\n","1. Firstly we prepare an initial state in one register. In parallel we prepare a uniform superposition state using Hadamard gates on some ancilla qubits. The number of ancilla qubits determines how precisely we can estimate the phase $\\theta$.\n","\n","2. Secondly we apply successive controlled $U$ gates. This has the effect of \"kicking back\" phases onto the ancilla qubits according to the eigenvalue equation above.\n","\n","3. Finally we apply the inverse Quantum Fourier Transform (QFT). This essentially plays the role of destructive interference, suppressing amplitudes from \"undesirable states\" and hopefully allowing us to measure a single outcome (or a small number of outcomes) with high probability.\n","\n","\n","There is some subtlety around the first point. The initial state used can be an exact eigenstate of $U$ however this may be difficult to prepare if we don't know the eigenvalues of $U$ in advance. Alternatively we could use an initial state that is a linear combination of eigenstates, as the phase estimation will project into the eigenspace of $U$."]},{"cell_type":"markdown","metadata":{},"source":["We also assume that we can implement $U$ with a quantum circuit. In chemistry applications $U$ could be of the form $U=e^{-iHt}$ where $H$ is the Hamiltonian of some system of interest. In the cannonical algorithm, the number of controlled unitaries we apply scales exponentially with the number of ancilla qubits. This allows more precision at the expense of a larger quantum circuit."]},{"cell_type":"markdown","metadata":{},"source":["## The Quantum Fourier Transform"]},{"cell_type":"markdown","metadata":{},"source":["Before considering the other parts of the QPE algorithm, lets focus on the Quantum Fourier Transform (QFT) subroutine.\n","\n","Mathematically, the QFT has the following action.\n","\n","$$\n","\\begin{equation}\n","QFT : |j\\rangle\\ \\longmapsto \\sum_{k=0}^{N - 1} e^{2 \\pi ijk/N}|k\\rangle, \\quad N= 2^k\n","\\end{equation}\n","$$\n","\n","This is essentially the Discrete Fourier transform except the input is a quantum state $|j\\rangle$.\n","\n","It is well known that the QFT can be implemented efficiently with a quantum circuit\n","\n","We can build the circuit for the $n$ qubit QFT using $n$ Hadamard gates $\\frac{n}{2}$ swap gates and $\\frac{n(n-1)}{2}$ controlled unitary rotations $\\text{CU1}$.\n","\n","$$\n"," \\begin{equation}\n"," CU1(\\phi) =\n"," \\begin{pmatrix}\n"," I & 0 \\\\\n"," 0 & U1(\\phi)\n"," \\end{pmatrix}\n"," \\,, \\quad\n","U1(\\phi) =\n"," \\begin{pmatrix}\n"," 1 & 0 \\\\\n"," 0 & e^{i \\phi}\n"," \\end{pmatrix}\n"," \\end{equation}\n","$$\n","\n","The circuit for the Quantum Fourier transform on three qubits is the following\n","\n","![](images/qft.png \"QFT Circuit\")\n","\n","We can build this circuit in `pytket` by adding gate operations manually:"]},{"cell_type":"markdown","metadata":{},"source":["lets build the QFT for three qubits"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import Circuit\n","from pytket.circuit.display import render_circuit_jupyter"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft3_circ = Circuit(3)\n","qft3_circ.H(0)\n","qft3_circ.CU1(0.5, 1, 0)\n","qft3_circ.CU1(0.25, 2, 0)\n","qft3_circ.H(1)\n","qft3_circ.CU1(0.5, 2, 1)\n","qft3_circ.H(2)\n","qft3_circ.SWAP(0, 2)\n","render_circuit_jupyter(qft3_circ)"]},{"cell_type":"markdown","metadata":{},"source":["We can generalise the quantum Fourier transform to $n$ qubits by iterating over the qubits as follows"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def build_qft_circuit(n_qubits: int) -> Circuit:\n"," circ = Circuit(n_qubits, name=\"QFT\")\n"," for i in range(n_qubits):\n"," circ.H(i)\n"," for j in range(i + 1, n_qubits):\n"," circ.CU1(1 / 2 ** (j - i), j, i)\n"," for k in range(0, n_qubits // 2):\n"," circ.SWAP(k, n_qubits - k - 1)\n"," return circ"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft4_circ: Circuit = build_qft_circuit(4)\n","render_circuit_jupyter(qft4_circ)"]},{"cell_type":"markdown","metadata":{},"source":["Now that we have the generalised circuit we can wrap it up in a `CircBox` which can then be added to another circuit as a subroutine."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import CircBox"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft4_box: CircBox = CircBox(qft4_circ)\n","qft_circ = Circuit(4).add_gate(qft4_box, [0, 1, 2, 3])\n","render_circuit_jupyter(qft_circ)"]},{"cell_type":"markdown","metadata":{},"source":["Note how the `CircBox` inherits the name `QFT` from the underlying circuit."]},{"cell_type":"markdown","metadata":{},"source":["Recall that in our phase estimation algorithm we need to use the inverse QFT.\n","\n","$$\n","\\begin{equation}\n","\\text{QFT}^† : \\sum_{k=0}^{N - 1} e^{2 \\pi ijk/N}|k\\rangle \\longmapsto |j\\rangle\\,, \\quad N= 2^k\n","\\end{equation}\n","$$\n","\n","\n","Now that we have the QFT circuit we can obtain the inverse by using `CircBox.dagger`. We can also verify that this is correct by inspecting the circuit inside with `CircBox.get_circuit()`."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["inv_qft4_box = qft4_box.dagger\n","render_circuit_jupyter(inv_qft4_box.get_circuit())"]},{"cell_type":"markdown","metadata":{},"source":["## The Controlled Unitary Operations"]},{"cell_type":"markdown","metadata":{},"source":["In the phase estimation algorithm we repeatedly perform controlled unitary operations. In the canonical variant, the number of controlled unitaries will be $2^m - 1$ where $m$ is the number of measurement qubits."]},{"cell_type":"markdown","metadata":{},"source":["The form of $U$ will vary depending on the application. For chemistry or condensed matter physics $U$ typically be the time evolution operator $U(t) = e^{- i H t}$ where $H$ is the problem Hamiltonian."]},{"cell_type":"markdown","metadata":{},"source":["Suppose that we had the following decomposition for $H$ in terms of Pauli strings $P_j$ and complex coefficients $\\alpha_j$.\n","\n","$$\n","\\begin{equation}\n","H = \\sum_j \\alpha_j P_j\\,, \\quad \\, P_j \\in \\{I, \\,X, \\,Y, \\,Z\\}^{\\otimes n}\n","\\end{equation}\n","$$\n","\n","Here Pauli strings refers to tensor products of Pauli operators. These strings form an orthonormal basis for $2^n \\times 2^n$ matrices."]},{"cell_type":"markdown","metadata":{},"source":["If we have a Hamiltonian in the form above, we can then implement $U(t)$ as a sequence of Pauli gadget circuits. We can do this with the [PauliExpBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.PauliExpBox) construct in pytket. For more on `PauliExpBox` see the [user manual](https://tket.quantinuum.com/user-manual/manual_circuit.html#pauli-exponential-boxes)."]},{"cell_type":"markdown","metadata":{},"source":["Once we have a circuit to implement our time evolution operator $U(t)$, we can construct the controlled $U(t)$ operations using [QControlBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.QControlBox). If our base unitary is a sequence of `PauliExpBox`(es) then there is some structure we can exploit to simplify our circuit. See this [blog post](https://tket.quantinuum.com/tket-blog/posts/controlled_gates/) on [ConjugationBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.ConjugationBox) for more."]},{"cell_type":"markdown","metadata":{},"source":["In what follows, we will just construct a simplified instance of QPE where the controlled unitaries are just $\\text{CU1}$ gates."]},{"cell_type":"markdown","metadata":{},"source":["## Putting it all together"]},{"cell_type":"markdown","metadata":{},"source":["We can now define a function to build our entire QPE circuit. We can make this function take a state preparation circuit and a unitary circuit as input as well. The function also has the number of measurement qubits as input which will determine the precision of our phase estimate."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import QControlBox"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def build_phase_est_circuit(\n"," n_measurement_qubits: int, state_prep_circuit: Circuit, unitary_circuit: Circuit\n",") -> Circuit:\n"," qpe_circ: Circuit = Circuit()\n"," n_state_prep_qubits = state_prep_circuit.n_qubits\n"," measurement_register = qpe_circ.add_q_register(\"m\", n_measurement_qubits)\n"," state_prep_register = qpe_circ.add_q_register(\"p\", n_state_prep_qubits)\n"," qpe_circ.add_circuit(state_prep_circuit, list(state_prep_register))\n","\n"," # Create a controlled unitary with a single control qubit\n"," unitary_circuit.name = \"U\"\n"," controlled_u_gate = QControlBox(CircBox(unitary_circuit), 1)\n","\n"," # Add Hadamard gates to every qubit in the measurement register\n"," for m_qubit in measurement_register:\n"," qpe_circ.H(m_qubit)\n","\n"," # Add all (2**n_measurement_qubits - 1) of the controlled unitaries sequentially\n"," for m_qubit in range(n_measurement_qubits):\n"," control_index = n_measurement_qubits - m_qubit - 1\n"," control_qubit = [measurement_register[control_index]]\n"," for _ in range(2**m_qubit):\n"," qpe_circ.add_qcontrolbox(\n"," controlled_u_gate, control_qubit + list(state_prep_register)\n"," )\n","\n"," # Finally, append the inverse qft and measure the qubits\n"," qft_box = CircBox(build_qft_circuit(n_measurement_qubits))\n"," inverse_qft_box = qft_box.dagger\n"," qpe_circ.add_circbox(inverse_qft_box, list(measurement_register))\n"," qpe_circ.measure_register(measurement_register, \"c\")\n"," return qpe_circ"]},{"cell_type":"markdown","metadata":{},"source":["## Phase Estimation with a Trivial Eigenstate\n","\n","Lets test our circuit construction by preparing a trivial $|1\\rangle$ eigenstate of the $\\text{U1}$ gate. We can then see if our phase estimation circuit returns the expected eigenvalue."]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","U1(\\phi)|1\\rangle = e^{i\\phi} = e^{2 \\pi i \\theta} \\implies \\theta = \\frac{\\phi}{2}\n","\\end{equation}\n","$$\n","\n","So we expect that our ideal phase $\\theta$ will be half the input angle $\\phi$ to our $U1$ gate."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["prep_circuit = Circuit(1).X(0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["input_angle = 0.73 # angle as number of half turns"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["unitary_circuit = Circuit(1).U1(input_angle, 0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qpe_circ_trivial = build_phase_est_circuit(\n"," 4, state_prep_circuit=prep_circuit, unitary_circuit=unitary_circuit\n",")"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["render_circuit_jupyter(qpe_circ_trivial)"]},{"cell_type":"markdown","metadata":{},"source":["Lets use the noiseless `AerBackend` simulator to run our phase estimation circuit."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.extensions.qiskit import AerBackend"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["backend = AerBackend()"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["compiled_circ = backend.get_compiled_circuit(qpe_circ_trivial)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_shots = 1000\n","result = backend.run_circuit(compiled_circ, n_shots)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(result.get_counts())"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.backends.backendresult import BackendResult\n","import matplotlib.pyplot as plt"]},{"cell_type":"markdown","metadata":{},"source":["plotting function for QPE Notebook"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def plot_qpe_results(\n"," sim_result: BackendResult,\n"," n_strings: int = 4,\n"," dark_mode: bool = False,\n"," y_limit: int = 1000,\n",") -> None:\n"," \"\"\"\n"," Plots results in a barchart given a BackendResult. the number of stings displayed\n"," can be specified with the n_strings argument.\n"," \"\"\"\n"," counts_dict = sim_result.get_counts()\n"," sorted_shots = counts_dict.most_common()\n"," n_most_common_strings = sorted_shots[:n_strings]\n"," x_axis_values = [str(entry[0]) for entry in n_most_common_strings] # basis states\n"," y_axis_values = [entry[1] for entry in n_most_common_strings] # counts\n"," if dark_mode:\n"," plt.style.use(\"dark_background\")\n"," fig = plt.figure()\n"," ax = fig.add_axes((0, 0, 0.75, 0.5))\n"," color_list = [\"orange\"] * (len(x_axis_values))\n"," ax.bar(\n"," x=x_axis_values,\n"," height=y_axis_values,\n"," color=color_list,\n"," )\n"," ax.set_title(label=\"Results\")\n"," plt.ylim([0, y_limit])\n"," plt.xlabel(\"Basis State\")\n"," plt.ylabel(\"Number of Shots\")\n"," plt.show()"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plot_qpe_results(result, y_limit=int(1.2 * n_shots))"]},{"cell_type":"markdown","metadata":{},"source":["As expected we see one outcome with high probability. Lets now extract our approximation of $\\theta$ from our output bitstrings.\n","\n","suppose the $j$ is an integer representation of our most commonly measured bitstring."]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","\\theta_{estimate} = \\frac{j}{N}\n","\\end{equation}\n","$$"]},{"cell_type":"markdown","metadata":{},"source":["Here $N = 2 ^n$ where $n$ is the number of measurement qubits."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.backends.backendresult import BackendResult"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def single_phase_from_backendresult(result: BackendResult) -> float:\n"," # Extract most common measurement outcome\n"," basis_state = result.get_counts().most_common()[0][0]\n"," bitstring = \"\".join([str(bit) for bit in basis_state])\n"," integer = int(bitstring, 2)\n","\n"," # Calculate theta estimate\n"," return integer / (2 ** len(bitstring))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["theta = single_phase_from_backendresult(result)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(theta)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(input_angle / 2)"]},{"cell_type":"markdown","metadata":{},"source":["Our output is close to half our input angle $\\phi$ as expected. Lets calculate our error."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["error = round(abs(input_angle - (2 * theta)), 3)\n","print(error)"]},{"cell_type":"markdown","metadata":{},"source":["## Suggestions for further reading\n","\n","In this notebook we have shown the canonical variant of quantum phase estimation. There are several other variants.\n","\n","Quantinuum paper on Bayesian phase estimation -> https://arxiv.org/pdf/2306.16608.pdf\n","Blog post on `ConjugationBox` -> https://tket.quantinuum.com/tket-blog/posts/controlled_gates/ - efficient circuits for controlled Pauli gadgets.\n","\n","As mentioned quantum phase estimation is a subroutine in Shor's algorithm. Read more about how phase estimation is used in period finding."]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} diff --git a/examples/python/pytket-qujax-classification.py b/examples/python/pytket-qujax-classification.py index 0a9d279a..3106c380 100644 --- a/examples/python/pytket-qujax-classification.py +++ b/examples/python/pytket-qujax-classification.py @@ -125,8 +125,16 @@ def param_and_x_to_probability(param, x_single): # \end{equation} # $$ # -# where $y'$ is a shot, $y$ is a data label and $\ell$ is some distance between bitstrings - here we simply set $\ell(0, 0) = \ell(1, 1) = 0$ and $\ell(0, 1) = \ell(1, 0) = 1$ (which coincides with the Hamming distance for this binary example). The full batch cost function is $C(b, w) = \frac1N \sum_{i=1}^N C(b, w, x_i, y_i)$. +# where $y'$ is a shot, $y$ is a data label and $\ell$ is some distance between bitstrings - here we simply set $\ell(0, 0) = \ell(1, 1) = 0$ and $\ell(0, 1) = \ell(1, 0) = 1$ (which coincides with the Hamming distance for this binary example). # +# The full batch cost function is +# +# $$ +# \begin{equation} +# C(b, w) = \frac1N \sum_{i=1}^N C(b,\, w,\, x_i,\, y_i). +# \end{equation} +# $$ + # Note that to calculate the cost function we need to evaluate the statetensor for every input point $x_i$. If the dataset becomes too large, we can easily minibatch. diff --git a/examples/pytket-qujax-classification.ipynb b/examples/pytket-qujax-classification.ipynb index 052e485d..78601918 100644 --- a/examples/pytket-qujax-classification.ipynb +++ b/examples/pytket-qujax-classification.ipynb @@ -1 +1 @@ -{"cells":[{"cell_type":"markdown","metadata":{},"source":["# Binary classification using pytket-qujax"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from jax import numpy as jnp, random, vmap, value_and_grad, jit\n","from pytket import Circuit\n","from pytket.circuit.display import render_circuit_jupyter\n","from pytket.extensions.qujax.qujax_convert import tk_to_qujax\n","import matplotlib.pyplot as plt"]},{"cell_type":"markdown","metadata":{},"source":["# Define the classification task\n","We'll try and learn a _donut_ binary classification function (i.e. a bivariate coordinate is labelled 1 if it is inside the donut and 0 if it is outside)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["inner_rad = 0.25\n","outer_rad = 0.75"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def classification_function(x, y):\n"," r = jnp.sqrt(x**2 + y**2)\n"," return jnp.where((r > inner_rad) * (r < outer_rad), 1, 0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["linsp = jnp.linspace(-1, 1, 1000)\n","Z = vmap(lambda x: vmap(lambda y: classification_function(x, y))(linsp))(linsp)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.contourf(linsp, linsp, Z, cmap=\"Purples\")"]},{"cell_type":"markdown","metadata":{},"source":["Now let's generate some data for our quantum circuit to learn from"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_data = 1000\n","x = random.uniform(random.PRNGKey(0), shape=(n_data, 2), minval=-1, maxval=1)\n","y = classification_function(x[:, 0], x[:, 1])"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.scatter(x[:, 0], x[:, 1], alpha=jnp.where(y, 1, 0.2), s=10)"]},{"cell_type":"markdown","metadata":{},"source":["# Quantum circuit time\n","We'll use a variant of data re-uploading [Pérez-Salinas et al](https://doi.org/10.22331/q-2020-02-06-226) to encode the input data, alongside some variational parameters within a quantum circuit classifier"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_qubits = 3\n","depth = 5"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["c = Circuit(n_qubits)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["for layer in range(depth):\n"," for qi in range(n_qubits):\n"," c.Rz(0.0, qi)\n"," c.Ry(0.0, qi)\n"," c.Rz(0.0, qi)\n"," if layer < (depth - 1):\n"," for qi in range(layer, layer + n_qubits - 1, 2):\n"," c.CZ(qi % n_qubits, (qi + 1) % n_qubits)\n"," c.add_barrier(range(n_qubits))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["render_circuit_jupyter(c)"]},{"cell_type":"markdown","metadata":{},"source":["We can use `pytket-qujax` to generate our angles-to-statetensor function."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["angles_to_st = tk_to_qujax(c)"]},{"cell_type":"markdown","metadata":{},"source":["We'll parameterise each angle as\n","\n","$$\n","\\begin{equation}\n","\\theta_k = b_k + w_k \\, x_k\n","\\end{equation}\n","$$\n","\n","where $b_k, w_k$ are variational parameters to be learnt and $x_k = x_0$ if $k$ even, $x_k = x_1$ if $k$ odd for a single bivariate input point $(x_0, x_1)$."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_angles = 3 * n_qubits * depth\n","n_params = 2 * n_angles"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def param_and_x_to_angles(param, x_single):\n"," biases = param[:n_angles]\n"," weights = param[n_angles:]\n"," weights_times_data = jnp.where(\n"," jnp.arange(n_angles) % 2 == 0, weights * x_single[0], weights * x_single[1]\n"," )\n"," angles = biases + weights_times_data\n"," return angles"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_and_x_to_st = lambda param, x_single: angles_to_st(\n"," param_and_x_to_angles(param, x_single)\n",")"]},{"cell_type":"markdown","metadata":{},"source":["We'll measure the first qubit only (if its 1 we label _donut_, if its 0 we label _not donut_)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def param_and_x_to_probability(param, x_single):\n"," st = param_and_x_to_st(param, x_single)\n"," all_probs = jnp.square(jnp.abs(st))\n"," first_qubit_probs = jnp.sum(all_probs, axis=range(1, n_qubits))\n"," return first_qubit_probs[1]"]},{"cell_type":"markdown","metadata":{},"source":["For binary classification, the likelihood for our full data set $(x_{1:N}, y_{1:N})$ is\n","\n","$$\n","\\begin{equation}\n","p(y_{1:N} \\mid b, w, x_{1:N}) = \\prod_{i=1}^N p(y_i \\mid b, w, x_i) = \\prod_{i=1}^N (1 - q_{(b,w)}(x_i))^{I[y_i = 0]}q_{(b,w)}(x_i)^{I[y_i = 1]},\n","\\end{equation}\n","$$\n","\n","where $q_{(b, w)}(x)$ is the probability the quantum circuit classifies input $x$ as donut given variational parameter vectors $(b, w)$. This gives log-likelihood\n","\n","$$\n","\\begin{equation}\n"," \\log p(y_{1:N} \\mid b, w, x_{1:N}) = \\sum_{i=1}^N I[y_i = 0] \\log(1 - q_{(b,w)}(x_i)) + I[y_i = 1] \\log q_{(b,w)}(x_i),\n","\\end{equation}\n","$$\n","\n","which we would like to maximise.\n","\n","Unfortunately, the log-likelihood **cannot** be approximated unbiasedly using shots, that is we can approximate $q_{(b,w)}(x_i)$ unbiasedly but not $\\log(q_{(b,w)}(x_i))$.\n","Note that in qujax simulations we can use the statetensor to calculate this exactly, but it is still good to keep in mind loss functions that can also be used with shots from a quantum device.\n","\n","Instead we can minimise an expected distance between shots and data\n","\n","$$\n","\\begin{equation}\n","C(b, w, x, y) = E_{p(y' \\mid q_{(b, w)}(x))}[\\ell(y', y)] = (1 - q_{(b, w)}(x)) \\ell(0, y) + q_{(b, w)}(x)\\ell(1, y),\n","\\end{equation}\n","$$\n","\n","where $y'$ is a shot, $y$ is a data label and $\\ell$ is some distance between bitstrings - here we simply set $\\ell(0, 0) = \\ell(1, 1) = 0$ and $\\ell(0, 1) = \\ell(1, 0) = 1$ (which coincides with the Hamming distance for this binary example). The full batch cost function is $C(b, w) = \\frac1N \\sum_{i=1}^N C(b, w, x_i, y_i)$.\n","\n","Note that to calculate the cost function we need to evaluate the statetensor for every input point $x_i$. If the dataset becomes too large, we can easily minibatch."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def param_to_cost(param):\n"," donut_probs = vmap(param_and_x_to_probability, in_axes=(None, 0))(param, x)\n"," costs = jnp.where(y, 1 - donut_probs, donut_probs)\n"," return costs.mean()"]},{"cell_type":"markdown","metadata":{},"source":["# Ready to descend some gradients?\n","We'll just use vanilla gradient descent here"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_cost_and_grad = jit(value_and_grad(param_to_cost))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_iter = 1000\n","stepsize = 1e-1\n","param = random.uniform(random.PRNGKey(1), shape=(n_params,), minval=0, maxval=2)\n","costs = jnp.zeros(n_iter)\n","for i in range(n_iter):\n"," cost, grad = param_to_cost_and_grad(param)\n"," costs = costs.at[i].set(cost)\n"," param = param - stepsize * grad\n"," print(i, \"Cost: \", cost, end=\"\\r\")"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.plot(costs)\n","plt.xlabel(\"Iteration\")\n","plt.ylabel(\"Cost\")"]},{"cell_type":"markdown","metadata":{},"source":["# Visualise trained classifier"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["linsp = jnp.linspace(-1, 1, 100)\n","Z = vmap(\n"," lambda a: vmap(lambda b: param_and_x_to_probability(param, jnp.array([a, b])))(\n"," linsp\n"," )\n",")(linsp)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.contourf(linsp, linsp, Z, cmap=\"Purples\", alpha=0.8)\n","circle_linsp = jnp.linspace(0, 2 * jnp.pi, 100)\n","plt.plot(inner_rad * jnp.cos(circle_linsp), inner_rad * jnp.sin(circle_linsp), c=\"red\")\n","plt.plot(outer_rad * jnp.cos(circle_linsp), outer_rad * jnp.sin(circle_linsp), c=\"red\")"]},{"cell_type":"markdown","metadata":{},"source":["Looks good, it has clearly grasped the donut shape. Sincerest apologies if you are now hungry! 🍩"]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} +{"cells":[{"cell_type":"markdown","metadata":{},"source":["# Binary classification using pytket-qujax"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from jax import numpy as jnp, random, vmap, value_and_grad, jit\n","from pytket import Circuit\n","from pytket.circuit.display import render_circuit_jupyter\n","from pytket.extensions.qujax.qujax_convert import tk_to_qujax\n","import matplotlib.pyplot as plt"]},{"cell_type":"markdown","metadata":{},"source":["# Define the classification task\n","We'll try and learn a _donut_ binary classification function (i.e. a bivariate coordinate is labelled 1 if it is inside the donut and 0 if it is outside)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["inner_rad = 0.25\n","outer_rad = 0.75"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def classification_function(x, y):\n"," r = jnp.sqrt(x**2 + y**2)\n"," return jnp.where((r > inner_rad) * (r < outer_rad), 1, 0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["linsp = jnp.linspace(-1, 1, 1000)\n","Z = vmap(lambda x: vmap(lambda y: classification_function(x, y))(linsp))(linsp)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.contourf(linsp, linsp, Z, cmap=\"Purples\")"]},{"cell_type":"markdown","metadata":{},"source":["Now let's generate some data for our quantum circuit to learn from"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_data = 1000\n","x = random.uniform(random.PRNGKey(0), shape=(n_data, 2), minval=-1, maxval=1)\n","y = classification_function(x[:, 0], x[:, 1])"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.scatter(x[:, 0], x[:, 1], alpha=jnp.where(y, 1, 0.2), s=10)"]},{"cell_type":"markdown","metadata":{},"source":["# Quantum circuit time\n","We'll use a variant of data re-uploading [Pérez-Salinas et al](https://doi.org/10.22331/q-2020-02-06-226) to encode the input data, alongside some variational parameters within a quantum circuit classifier"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_qubits = 3\n","depth = 5"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["c = Circuit(n_qubits)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["for layer in range(depth):\n"," for qi in range(n_qubits):\n"," c.Rz(0.0, qi)\n"," c.Ry(0.0, qi)\n"," c.Rz(0.0, qi)\n"," if layer < (depth - 1):\n"," for qi in range(layer, layer + n_qubits - 1, 2):\n"," c.CZ(qi % n_qubits, (qi + 1) % n_qubits)\n"," c.add_barrier(range(n_qubits))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["render_circuit_jupyter(c)"]},{"cell_type":"markdown","metadata":{},"source":["We can use `pytket-qujax` to generate our angles-to-statetensor function."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["angles_to_st = tk_to_qujax(c)"]},{"cell_type":"markdown","metadata":{},"source":["We'll parameterise each angle as\n","\n","$$\n","\\begin{equation}\n","\\theta_k = b_k + w_k \\, x_k\n","\\end{equation}\n","$$\n","\n","where $b_k, w_k$ are variational parameters to be learnt and $x_k = x_0$ if $k$ even, $x_k = x_1$ if $k$ odd for a single bivariate input point $(x_0, x_1)$."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_angles = 3 * n_qubits * depth\n","n_params = 2 * n_angles"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def param_and_x_to_angles(param, x_single):\n"," biases = param[:n_angles]\n"," weights = param[n_angles:]\n"," weights_times_data = jnp.where(\n"," jnp.arange(n_angles) % 2 == 0, weights * x_single[0], weights * x_single[1]\n"," )\n"," angles = biases + weights_times_data\n"," return angles"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_and_x_to_st = lambda param, x_single: angles_to_st(\n"," param_and_x_to_angles(param, x_single)\n",")"]},{"cell_type":"markdown","metadata":{},"source":["We'll measure the first qubit only (if its 1 we label _donut_, if its 0 we label _not donut_)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def param_and_x_to_probability(param, x_single):\n"," st = param_and_x_to_st(param, x_single)\n"," all_probs = jnp.square(jnp.abs(st))\n"," first_qubit_probs = jnp.sum(all_probs, axis=range(1, n_qubits))\n"," return first_qubit_probs[1]"]},{"cell_type":"markdown","metadata":{},"source":["For binary classification, the likelihood for our full data set $(x_{1:N}, y_{1:N})$ is\n","\n","$$\n","\\begin{equation}\n","p(y_{1:N} \\mid b, w, x_{1:N}) = \\prod_{i=1}^N p(y_i \\mid b, w, x_i) = \\prod_{i=1}^N (1 - q_{(b,w)}(x_i))^{I[y_i = 0]}q_{(b,w)}(x_i)^{I[y_i = 1]},\n","\\end{equation}\n","$$\n","\n","where $q_{(b, w)}(x)$ is the probability the quantum circuit classifies input $x$ as donut given variational parameter vectors $(b, w)$. This gives log-likelihood\n","\n","$$\n","\\begin{equation}\n"," \\log p(y_{1:N} \\mid b, w, x_{1:N}) = \\sum_{i=1}^N I[y_i = 0] \\log(1 - q_{(b,w)}(x_i)) + I[y_i = 1] \\log q_{(b,w)}(x_i),\n","\\end{equation}\n","$$\n","\n","which we would like to maximise.\n","\n","Unfortunately, the log-likelihood **cannot** be approximated unbiasedly using shots, that is we can approximate $q_{(b,w)}(x_i)$ unbiasedly but not $\\log(q_{(b,w)}(x_i))$.\n","Note that in qujax simulations we can use the statetensor to calculate this exactly, but it is still good to keep in mind loss functions that can also be used with shots from a quantum device.\n","\n","Instead we can minimise an expected distance between shots and data\n","\n","$$\n","\\begin{equation}\n","C(b, w, x, y) = E_{p(y' \\mid q_{(b, w)}(x))}[\\ell(y', y)] = (1 - q_{(b, w)}(x)) \\ell(0, y) + q_{(b, w)}(x)\\ell(1, y),\n","\\end{equation}\n","$$\n","\n","where $y'$ is a shot, $y$ is a data label and $\\ell$ is some distance between bitstrings - here we simply set $\\ell(0, 0) = \\ell(1, 1) = 0$ and $\\ell(0, 1) = \\ell(1, 0) = 1$ (which coincides with the Hamming distance for this binary example).\n","\n"," The full batch cost function is\n","\n","$$\n","\\begin{equation}\n"," C(b, w) = \\frac1N \\sum_{i=1}^N C(b,\\, w,\\, x_i,\\, y_i).\n","\\end{equation}\n","$$"]},{"cell_type":"markdown","metadata":{},"source":["Note that to calculate the cost function we need to evaluate the statetensor for every input point $x_i$. If the dataset becomes too large, we can easily minibatch."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def param_to_cost(param):\n"," donut_probs = vmap(param_and_x_to_probability, in_axes=(None, 0))(param, x)\n"," costs = jnp.where(y, 1 - donut_probs, donut_probs)\n"," return costs.mean()"]},{"cell_type":"markdown","metadata":{},"source":["# Ready to descend some gradients?\n","We'll just use vanilla gradient descent here"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_cost_and_grad = jit(value_and_grad(param_to_cost))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_iter = 1000\n","stepsize = 1e-1\n","param = random.uniform(random.PRNGKey(1), shape=(n_params,), minval=0, maxval=2)\n","costs = jnp.zeros(n_iter)\n","for i in range(n_iter):\n"," cost, grad = param_to_cost_and_grad(param)\n"," costs = costs.at[i].set(cost)\n"," param = param - stepsize * grad\n"," print(i, \"Cost: \", cost, end=\"\\r\")"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.plot(costs)\n","plt.xlabel(\"Iteration\")\n","plt.ylabel(\"Cost\")"]},{"cell_type":"markdown","metadata":{},"source":["# Visualise trained classifier"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["linsp = jnp.linspace(-1, 1, 100)\n","Z = vmap(\n"," lambda a: vmap(lambda b: param_and_x_to_probability(param, jnp.array([a, b])))(\n"," linsp\n"," )\n",")(linsp)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.contourf(linsp, linsp, Z, cmap=\"Purples\", alpha=0.8)\n","circle_linsp = jnp.linspace(0, 2 * jnp.pi, 100)\n","plt.plot(inner_rad * jnp.cos(circle_linsp), inner_rad * jnp.sin(circle_linsp), c=\"red\")\n","plt.plot(outer_rad * jnp.cos(circle_linsp), outer_rad * jnp.sin(circle_linsp), c=\"red\")"]},{"cell_type":"markdown","metadata":{},"source":["Looks good, it has clearly grasped the donut shape. Sincerest apologies if you are now hungry! 🍩"]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} diff --git a/examples/pytket-qujax_qaoa.ipynb b/examples/pytket-qujax_qaoa.ipynb index a64e5812..b50fd42d 100644 --- a/examples/pytket-qujax_qaoa.ipynb +++ b/examples/pytket-qujax_qaoa.ipynb @@ -1 +1 @@ -{"cells":[{"cell_type":"markdown","metadata":{},"source":["# Symbolic circuits with `qujax` and `pytket-qujax`
\n","In this notebook we will show how to manipulate symbolic circuits with the `pytket-qujax` extension. In particular, we will consider a QAOA and an Ising Hamiltonian."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket import Circuit\n","from pytket.circuit.display import render_circuit_jupyter\n","from jax import numpy as jnp, random, value_and_grad, jit\n","from sympy import Symbol\n","import matplotlib.pyplot as plt"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["import qujax\n","from pytket.extensions.qujax import tk_to_qujax"]},{"cell_type":"markdown","metadata":{},"source":["# QAOA\n","The Quantum Approximate Optimization Algorithm (QAOA), first introduced by [Farhi et al.](https://arxiv.org/pdf/1411.4028.pdf), is a quantum variational algorithm used to solve optimization problems. It consists of a unitary $U(\\beta, \\gamma)$ formed by alternate repetitions of $U(\\beta)=e^{-i\\beta H_B}$ and $U(\\gamma)=e^{-i\\gamma H_P}$, where $H_B$ is the mixing Hamiltonian and $H_P$ the problem Hamiltonian. The goal is to find the optimal parameters that minimize $H_P$.\n","Given a depth $d$, the expression of the final unitary is $U(\\beta, \\gamma) = U(\\beta_d)U(\\gamma_d)\\cdots U(\\beta_1)U(\\gamma_1)$. Notice that for each repetition the parameters are different.\n","\n","## Problem Hamiltonian\n","QAOA uses a problem dependent ansatz. Therefore, we first need to know the problem that we want to solve. In this case we will consider an Ising Hamiltonian with only $Z$ interactions. Given a set of pairs (or qubit indices) $E$, the problem Hamiltonian will be:\n","\n","$$\n","\\begin{equation}\n","H_P = \\sum_{(i, j) \\in E}\\alpha_{ij}Z_iZ_j,\n","\\end{equation}\n","$$\n","\n","where $\\alpha_{ij}$ are the coefficients.\n","Let's build our problem Hamiltonian with random coefficients and a set of pairs for a given number of qubits:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_qubits = 4\n","hamiltonian_qubit_inds = [(0, 1), (1, 2), (0, 2), (1, 3)]\n","hamiltonian_gates = [[\"Z\", \"Z\"]] * (len(hamiltonian_qubit_inds))"]},{"cell_type":"markdown","metadata":{},"source":["Notice that in order to use the random package from jax we first need to define a seeded key"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["seed = 13\n","key = random.PRNGKey(seed)\n","coefficients = random.uniform(key, shape=(len(hamiltonian_qubit_inds),))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(\"Gates:\\t\", hamiltonian_gates)\n","print(\"Qubits:\\t\", hamiltonian_qubit_inds)\n","print(\"Coefficients:\\t\", coefficients)"]},{"cell_type":"markdown","metadata":{},"source":["## Variational Circuit\n","Before constructing the circuit, we still need to select the mixing Hamiltonian. In our case, we will be using $X$ gates in each qubit, so $H_B = \\sum_{i=1}^{n}X_i$, where $n$ is the number of qubits. Notice that the unitary $U(\\beta)$, given this mixing Hamiltonian, is an $X$ rotation in each qubit with angle $\\beta$.\n","As for the unitary corresponding to the problem Hamiltonian, $U(\\gamma)$, it has the following form:\n","\n","$$\n","\\begin{equation}\n","U(\\gamma)=\\prod_{(i, j) \\in E}e^{-i\\gamma\\alpha_{ij}Z_i Z_j}\n","\\end{equation}\n","$$\n","\n","The operation $e^{-i\\gamma\\alpha_{ij}Z_iZ_j}$ can be performed using two CNOT gates with qubit $i$ as control and qubit $j$ as target and a $Z$ rotation in qubit $j$ in between them, with angle $\\gamma\\alpha_{ij}$.\n","Finally, the initial state used, in general, with the QAOA is an equal superposition of all the basis states. This can be achieved adding a first layer of Hadamard gates in each qubit at the beginning of the circuit."]},{"cell_type":"markdown","metadata":{},"source":["With all the building blocks, let's construct the symbolic circuit using tket. Notice that in order to define the parameters, we use the ```Symbol``` object from the `sympy` package. More info can be found in this [documentation](https://cqcl.github.io/pytket/manual/manual_circuit.html#symbolic-circuits). In order to later convert the circuit to qujax, we need to return the list of symbolic parameters as well."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def qaoa_circuit(n_qubits, depth):\n"," circuit = Circuit(n_qubits)\n"," p_keys = []\n","\n"," # Initial State\n"," for i in range(n_qubits):\n"," circuit.H(i)\n"," for d in range(depth):\n"," # Hamiltonian unitary\n"," gamma_d = Symbol(f\"γ_{d}\")\n"," for index in range(len(hamiltonian_qubit_inds)):\n"," pair = hamiltonian_qubit_inds[index]\n"," coef = coefficients[index]\n"," circuit.CX(pair[0], pair[1])\n"," circuit.Rz(gamma_d * coef, pair[1])\n"," circuit.CX(pair[0], pair[1])\n"," circuit.add_barrier(range(0, n_qubits))\n"," p_keys.append(gamma_d)\n","\n"," # Mixing unitary\n"," beta_d = Symbol(f\"β_{d}\")\n"," for i in range(n_qubits):\n"," circuit.Rx(beta_d, i)\n"," p_keys.append(beta_d)\n"," return circuit, p_keys"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["depth = 3\n","circuit, keys = qaoa_circuit(n_qubits, depth)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["keys"]},{"cell_type":"markdown","metadata":{},"source":["Let's check the circuit:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["render_circuit_jupyter(circuit)"]},{"cell_type":"markdown","metadata":{},"source":["# Now for `qujax`
\n","The `pytket.extensions.qujax.tk_to_qujax` function will generate a parameters -> statetensor function for us. However, in order to convert a symbolic circuit we first need to define the `symbol_map`. This object maps each symbol key to their corresponding index. In our case, since the object `keys` contains the symbols in the correct order, we can simply construct the dictionary as follows:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["symbol_map = {keys[i]: i for i in range(len(keys))}"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["symbol_map"]},{"cell_type":"markdown","metadata":{},"source":["Then, we invoke the `tk_to_qujax` with both the circuit and the symbolic map."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_st = tk_to_qujax(circuit, symbol_map=symbol_map)"]},{"cell_type":"markdown","metadata":{},"source":["And we also construct the expectation map using the problem Hamiltonian via qujax:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["st_to_expectation = qujax.get_statetensor_to_expectation_func(\n"," hamiltonian_gates, hamiltonian_qubit_inds, coefficients\n",")"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_expectation = lambda param: st_to_expectation(param_to_st(param))"]},{"cell_type":"markdown","metadata":{},"source":["# Training process
\n","We construct a function that, given a parameter vector, returns the value of the cost function and the gradient.
\n","We also `jit` to avoid recompilation, this means that the expensive `cost_and_grad` function is compiled once into a very fast XLA (C++) function which is then executed at each iteration. Alternatively, we could get the same speedup by replacing our `for` loop with `jax.lax.scan`. You can read more about JIT compilation in the [JAX documentation](https://jax.readthedocs.io/en/latest/jax-101/02-jitting.html)."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["cost_and_grad = jit(value_and_grad(param_to_expectation))"]},{"cell_type":"markdown","metadata":{},"source":["For the training process we'll use vanilla gradient descent with a constant stepsize:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["seed = 123\n","key = random.PRNGKey(seed)\n","init_param = random.uniform(key, shape=(len(symbol_map),))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_steps = 150\n","stepsize = 0.01"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param = init_param"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["cost_vals = jnp.zeros(n_steps)\n","cost_vals = cost_vals.at[0].set(param_to_expectation(init_param))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["for step in range(1, n_steps):\n"," cost_val, cost_grad = cost_and_grad(param)\n"," cost_vals = cost_vals.at[step].set(cost_val)\n"," param = param - stepsize * cost_grad\n"," print(\"Iteration:\", step, \"\\tCost:\", cost_val, end=\"\\r\")"]},{"cell_type":"markdown","metadata":{},"source":["Let's visualise the gradient descent"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.plot(cost_vals)\n","plt.xlabel(\"Iteration\")\n","plt.ylabel(\"Cost\")"]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} +{"cells":[{"cell_type":"markdown","metadata":{},"source":["# Symbolic circuits with `qujax` and `pytket-qujax`\n","In this notebook we will show how to manipulate symbolic circuits with the `pytket-qujax` extension. In particular, we will consider a QAOA and an Ising Hamiltonian."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket import Circuit\n","from pytket.circuit.display import render_circuit_jupyter\n","from jax import numpy as jnp, random, value_and_grad, jit\n","from sympy import Symbol\n","import matplotlib.pyplot as plt"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["import qujax\n","from pytket.extensions.qujax import tk_to_qujax"]},{"cell_type":"markdown","metadata":{},"source":["# QAOA\n","The Quantum Approximate Optimization Algorithm (QAOA), first introduced by [Farhi et al.](https://arxiv.org/pdf/1411.4028.pdf), is a quantum variational algorithm used to solve optimization problems. It consists of a unitary $U(\\beta, \\gamma)$ formed by alternate repetitions of $U(\\beta)=e^{-i\\beta H_B}$ and $U(\\gamma)=e^{-i\\gamma H_P}$, where $H_B$ is the mixing Hamiltonian and $H_P$ the problem Hamiltonian. The goal is to find the optimal parameters that minimize $H_P$.\n","Given a depth $d$, the expression of the final unitary is $U(\\beta, \\gamma) = U(\\beta_d)U(\\gamma_d)\\cdots U(\\beta_1)U(\\gamma_1)$. Notice that for each repetition the parameters are different.\n","\n","## Problem Hamiltonian\n","QAOA uses a problem dependent ansatz. Therefore, we first need to know the problem that we want to solve. In this case we will consider an Ising Hamiltonian with only $Z$ interactions. Given a set of pairs (or qubit indices) $E$, the problem Hamiltonian will be:\n","\n","$$\n","\\begin{equation}\n","H_P = \\sum_{(i, j) \\in E}\\alpha_{ij}Z_iZ_j,\n","\\end{equation}\n","$$\n","\n","where $\\alpha_{ij}$ are the coefficients.\n","Let's build our problem Hamiltonian with random coefficients and a set of pairs for a given number of qubits:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_qubits = 4\n","hamiltonian_qubit_inds = [(0, 1), (1, 2), (0, 2), (1, 3)]\n","hamiltonian_gates = [[\"Z\", \"Z\"]] * (len(hamiltonian_qubit_inds))"]},{"cell_type":"markdown","metadata":{},"source":["Notice that in order to use the random package from jax we first need to define a seeded key"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["seed = 13\n","key = random.PRNGKey(seed)\n","coefficients = random.uniform(key, shape=(len(hamiltonian_qubit_inds),))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(\"Gates:\\t\", hamiltonian_gates)\n","print(\"Qubits:\\t\", hamiltonian_qubit_inds)\n","print(\"Coefficients:\\t\", coefficients)"]},{"cell_type":"markdown","metadata":{},"source":["## Variational Circuit\n","Before constructing the circuit, we still need to select the mixing Hamiltonian. In our case, we will be using $X$ gates in each qubit, so $H_B = \\sum_{i=1}^{n}X_i$, where $n$ is the number of qubits. Notice that the unitary $U(\\beta)$, given this mixing Hamiltonian, is an $X$ rotation in each qubit with angle $\\beta$.\n","As for the unitary corresponding to the problem Hamiltonian, $U(\\gamma)$, it has the following form:\n","\n","$$\n","\\begin{equation}\n","U(\\gamma)=\\prod_{(i, j) \\in E}e^{-i\\gamma\\alpha_{ij}Z_i Z_j}\n","\\end{equation}\n","$$\n","\n","The operation $e^{-i\\gamma\\alpha_{ij}Z_iZ_j}$ can be performed using two CNOT gates with qubit $i$ as control and qubit $j$ as target and a $Z$ rotation in qubit $j$ in between them, with angle $\\gamma\\alpha_{ij}$.\n","Finally, the initial state used, in general, with the QAOA is an equal superposition of all the basis states. This can be achieved adding a first layer of Hadamard gates in each qubit at the beginning of the circuit."]},{"cell_type":"markdown","metadata":{},"source":["With all the building blocks, let's construct the symbolic circuit using tket. Notice that in order to define the parameters, we use the ```Symbol``` object from the `sympy` package. More info can be found in this [documentation](https://cqcl.github.io/pytket/manual/manual_circuit.html#symbolic-circuits). In order to later convert the circuit to qujax, we need to return the list of symbolic parameters as well."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def qaoa_circuit(n_qubits, depth):\n"," circuit = Circuit(n_qubits)\n"," p_keys = []\n","\n"," # Initial State\n"," for i in range(n_qubits):\n"," circuit.H(i)\n"," for d in range(depth):\n"," # Hamiltonian unitary\n"," gamma_d = Symbol(f\"γ_{d}\")\n"," for index in range(len(hamiltonian_qubit_inds)):\n"," pair = hamiltonian_qubit_inds[index]\n"," coef = coefficients[index]\n"," circuit.CX(pair[0], pair[1])\n"," circuit.Rz(gamma_d * coef, pair[1])\n"," circuit.CX(pair[0], pair[1])\n"," circuit.add_barrier(range(0, n_qubits))\n"," p_keys.append(gamma_d)\n","\n"," # Mixing unitary\n"," beta_d = Symbol(f\"β_{d}\")\n"," for i in range(n_qubits):\n"," circuit.Rx(beta_d, i)\n"," p_keys.append(beta_d)\n"," return circuit, p_keys"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["depth = 3\n","circuit, keys = qaoa_circuit(n_qubits, depth)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["keys"]},{"cell_type":"markdown","metadata":{},"source":["Let's check the circuit:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["render_circuit_jupyter(circuit)"]},{"cell_type":"markdown","metadata":{},"source":["# Now for `qujax`\n","The `pytket.extensions.qujax.tk_to_qujax` function will generate a parameters -> statetensor function for us. However, in order to convert a symbolic circuit we first need to define the `symbol_map`. This object maps each symbol key to their corresponding index. In our case, since the object `keys` contains the symbols in the correct order, we can simply construct the dictionary as follows:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["symbol_map = {keys[i]: i for i in range(len(keys))}"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["symbol_map"]},{"cell_type":"markdown","metadata":{},"source":["Then, we invoke the `tk_to_qujax` with both the circuit and the symbolic map."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_st = tk_to_qujax(circuit, symbol_map=symbol_map)"]},{"cell_type":"markdown","metadata":{},"source":["And we also construct the expectation map using the problem Hamiltonian via qujax:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["st_to_expectation = qujax.get_statetensor_to_expectation_func(\n"," hamiltonian_gates, hamiltonian_qubit_inds, coefficients\n",")"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_expectation = lambda param: st_to_expectation(param_to_st(param))"]},{"cell_type":"markdown","metadata":{},"source":["# Training process\n","We construct a function that, given a parameter vector, returns the value of the cost function and the gradient.\n","We also `jit` to avoid recompilation, this means that the expensive `cost_and_grad` function is compiled once into a very fast XLA (C++) function which is then executed at each iteration. Alternatively, we could get the same speedup by replacing our `for` loop with `jax.lax.scan`. You can read more about JIT compilation in the [JAX documentation](https://jax.readthedocs.io/en/latest/jax-101/02-jitting.html)."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["cost_and_grad = jit(value_and_grad(param_to_expectation))"]},{"cell_type":"markdown","metadata":{},"source":["For the training process we'll use vanilla gradient descent with a constant stepsize:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["seed = 123\n","key = random.PRNGKey(seed)\n","init_param = random.uniform(key, shape=(len(symbol_map),))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_steps = 150\n","stepsize = 0.01"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param = init_param"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["cost_vals = jnp.zeros(n_steps)\n","cost_vals = cost_vals.at[0].set(param_to_expectation(init_param))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["for step in range(1, n_steps):\n"," cost_val, cost_grad = cost_and_grad(param)\n"," cost_vals = cost_vals.at[step].set(cost_val)\n"," param = param - stepsize * cost_grad\n"," print(\"Iteration:\", step, \"\\tCost:\", cost_val, end=\"\\r\")"]},{"cell_type":"markdown","metadata":{},"source":["Let's visualise the gradient descent"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.plot(cost_vals)\n","plt.xlabel(\"Iteration\")\n","plt.ylabel(\"Cost\")"]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} From ee9bee3be7ccd69a12dd528533a9a7cb1435da68 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Tue, 7 Nov 2023 13:03:57 +0000 Subject: [PATCH 41/51] remove outdated info from ansatz sequencing example --- examples/ansatz_sequence_example.ipynb | 2 +- examples/python/ansatz_sequence_example.py | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/examples/ansatz_sequence_example.ipynb b/examples/ansatz_sequence_example.ipynb index 1ef80ee4..cd250ada 100644 --- a/examples/ansatz_sequence_example.ipynb +++ b/examples/ansatz_sequence_example.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Ansatz sequencing"]}, {"cell_type": "markdown", "metadata": {}, "source": ["When performing variational algorithms like VQE, one common approach to generating circuit ans\u00e4tze is to take an operator $U$ representing excitations and use this to act on a reference state $\\lvert \\phi_0 \\rangle$. One such ansatz is the Unitary Coupled Cluster ansatz. Each excitation, indexed by $j$, within $U$ is given a real coefficient $a_j$ and a parameter $t_j$, such that $U = e^{i \\sum_j \\sum_k a_j t_j P_{jk}}$, where $P_{jk} \\in \\{I, X, Y, Z \\}^{\\otimes n}$. The exact form is dependent on the chosen qubit encoding. This excitation gives us a variational state $\\lvert \\psi (t) \\rangle = U(t) \\lvert \\phi_0 \\rangle$. The operator $U$ must be Trotterised, to give a product of Pauli exponentials, and converted into native quantum gates to create the ansatz circuit.
\n", "
\n", "This notebook will describe how to use an advanced feature of `pytket` to enable automated circuit synthesis for $U$ and reduce circuit depth dramatically.
\n", "
\n", "We must create a `pytket` `QubitPauliOperator`, which represents such an operator $U$, and contains a dictionary from Pauli string $P_{jk}$ to symbolic expression. Here, we make a mock operator ourselves, which resembles the UCCSD excitation operator for the $\\mathrm{H}_2$ molecule using the Jordan-Wigner qubit encoding. In the future, operator generation will be handled automatically using CQC's upcoming software for enterprise quantum chemistry, EUMEN. We also offer conversion to and from the `OpenFermion` `QubitOperator` class, although at the time of writing a `QubitOperator` cannot handle arbitrary symbols.
\n", "
\n", "First, we create a series of `QubitPauliString` objects, which represent each $P_{jk}$."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.circuit import Qubit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["q = [Qubit(i) for i in range(4)]\n", "qps0 = QubitPauliString([q[0], q[1], q[2]], [Pauli.Y, Pauli.Z, Pauli.X])\n", "qps1 = QubitPauliString([q[0], q[1], q[2]], [Pauli.X, Pauli.Z, Pauli.Y])\n", "qps2 = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.X, Pauli.Y])\n", "qps3 = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.Y, Pauli.X])\n", "qps4 = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.X, Pauli.X])\n", "qps5 = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.X, Pauli.X])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now, create some symbolic expressions for the $a_j t_j$ terms."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import fresh_symbol"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["symbol1 = fresh_symbol(\"s0\")\n", "expr1 = 1.2 * symbol1\n", "symbol2 = fresh_symbol(\"s1\")\n", "expr2 = -0.3 * symbol2"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can now create our `QubitPauliOperator`."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils import QubitPauliOperator"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["dict1 = dict((string, expr1) for string in (qps0, qps1))\n", "dict2 = dict((string, expr2) for string in (qps2, qps3, qps4, qps5))\n", "operator = QubitPauliOperator({**dict1, **dict2})\n", "print(operator)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now we can let `pytket` sequence the terms in this operator for us, using a selection of strategies. First, we will create a `Circuit` to generate an example reference state, and then use the `gen_term_sequence_circuit` method to append the Pauli exponentials."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import Circuit\n", "from pytket.utils import gen_term_sequence_circuit\n", "from pytket.partition import PauliPartitionStrat, GraphColourMethod"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["reference_circ = Circuit(4).X(1).X(3)\n", "ansatz_circuit = gen_term_sequence_circuit(\n", " operator, reference_circ, PauliPartitionStrat.CommutingSets, GraphColourMethod.Lazy\n", ")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This method works by generating a graph of Pauli exponentials and performing graph colouring. Here we have chosen to partition the terms so that exponentials which commute are gathered together, and we have done so using a lazy, greedy graph colouring method.
\n", "
\n", "Alternatively, we could have used the `PauliPartitionStrat.NonConflictingSets`, which puts Pauli exponentials together so that they only require single-qubit gates to be converted into the form $e^{i \\alpha Z \\otimes Z \\otimes ... \\otimes Z}$. This strategy is primarily useful for measurement reduction, a different problem.
\n", "
\n", "We could also have used the `GraphColourMethod.LargestFirst`, which still uses a greedy method, but builds the full graph and iterates through the vertices in descending order of arity. We recommend playing around with the options, but we typically find that the combination of `CommutingSets` and `Lazy` allows the best optimisation.
\n", "
\n", "In general, not all of our exponentials will commute, so the semantics of our circuit depend on the order of our sequencing. As a result, it is important for us to be able to inspect the order we have produced. `pytket` provides functionality to enable this. Each set of commuting exponentials is put into a `CircBox`, which lets us inspect the partitoning."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import OpType"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["for command in ansatz_circuit:\n", " if command.op.type == OpType.CircBox:\n", " print(\"New CircBox:\")\n", " for pauli_exp in command.op.get_circuit():\n", " print(\n", " \" {} {} {}\".format(\n", " pauli_exp, pauli_exp.op.get_paulis(), pauli_exp.op.get_phase()\n", " )\n", " )\n", " else:\n", " print(\"Native gate: {}\".format(command))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can convert this circuit into basic gates using a `pytket` `Transform`. This acts in place on the circuit to do rewriting, for gate translation and optimisation. We will start off with a naive decomposition."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.transform import Transform"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["naive_circuit = ansatz_circuit.copy()\n", "Transform.DecomposeBoxes().apply(naive_circuit)\n", "print(naive_circuit.get_commands())"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is a jumble of one- and two-qubit gates. We can get some relevant circuit metrics out:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"Naive CX Depth: {}\".format(naive_circuit.depth_by_type(OpType.CX)))\n", "print(\"Naive CX Count: {}\".format(naive_circuit.n_gates_of_type(OpType.CX)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These metrics can be improved upon significantly by smart compilation. A `Transform` exists precisely for this purpose:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.transform import PauliSynthStrat, CXConfigType"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["smart_circuit = ansatz_circuit.copy()\n", "Transform.UCCSynthesis(PauliSynthStrat.Sets, CXConfigType.Tree).apply(smart_circuit)\n", "print(\"Smart CX Depth: {}\".format(smart_circuit.depth_by_type(OpType.CX)))\n", "print(\"Smart CX Count: {}\".format(smart_circuit.n_gates_of_type(OpType.CX)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This `Transform` takes in a `Circuit` with the structure specified above: some arbitrary gates for the reference state, along with several `CircBox` gates containing `PauliExpBox` gates.
\n", "
\n", "We have chosen `PauliSynthStrat.Sets` and `CXConfigType.Tree`. The `PauliSynthStrat` dictates the method for decomposing multiple adjacent Pauli exponentials into basic gates, while the `CXConfigType` dictates the structure of adjacent CX gates.
\n", "
\n", "If we choose a different combination of strategies, we can produce a different output circuit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["last_circuit = ansatz_circuit.copy()\n", "Transform.UCCSynthesis(PauliSynthStrat.Individual, CXConfigType.Snake).apply(\n", " last_circuit\n", ")\n", "print(last_circuit.get_commands())"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"Last CX Depth: {}\".format(last_circuit.depth_by_type(OpType.CX)))\n", "print(\"Last CX Count: {}\".format(last_circuit.n_gates_of_type(OpType.CX)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Other than some single-qubit Cliffords we acquired via synthesis, you can check that this gives us the same circuit structure as our `Transform.DecomposeBoxes` method! It is a suboptimal synthesis method.
\n", "
\n", "As with the `gen_term_sequence` method, we recommend playing around with the arguments and seeing what circuits come out. Typically we find that `PauliSynthStrat.Sets` and `CXConfigType.Tree` work the best, although routing can affect this somewhat."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Ansatz sequencing"]}, {"cell_type": "markdown", "metadata": {}, "source": ["When performing variational algorithms like VQE, one common approach to generating circuit ans\u00e4tze is to take an operator $U$ representing excitations and use this to act on a reference state $\\lvert \\phi_0 \\rangle$. One such ansatz is the Unitary Coupled Cluster ansatz. Each excitation, indexed by $j$, within $U$ is given a real coefficient $a_j$ and a parameter $t_j$, such that $U = e^{i \\sum_j \\sum_k a_j t_j P_{jk}}$, where $P_{jk} \\in \\{I, X, Y, Z \\}^{\\otimes n}$. The exact form is dependent on the chosen qubit encoding. This excitation gives us a variational state $\\lvert \\psi (t) \\rangle = U(t) \\lvert \\phi_0 \\rangle$. The operator $U$ must be Trotterised, to give a product of Pauli exponentials, and converted into native quantum gates to create the ansatz circuit.
\n", "
\n", "This notebook will describe how to use an advanced feature of `pytket` to enable automated circuit synthesis for $U$ and reduce circuit depth dramatically.
\n", "
\n", "We must create a `pytket` `QubitPauliOperator`, which represents such an operator $U$, and contains a dictionary from Pauli string $P_{jk}$ to symbolic expression. Here, we make a mock operator ourselves, which resembles the UCCSD excitation operator for the $\\mathrm{H}_2$ molecule using the Jordan-Wigner qubit encoding.
\n", "
\n", "First, we create a series of `QubitPauliString` objects, which represent each $P_{jk}$."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.circuit import Qubit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["q = [Qubit(i) for i in range(4)]\n", "qps0 = QubitPauliString([q[0], q[1], q[2]], [Pauli.Y, Pauli.Z, Pauli.X])\n", "qps1 = QubitPauliString([q[0], q[1], q[2]], [Pauli.X, Pauli.Z, Pauli.Y])\n", "qps2 = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.X, Pauli.Y])\n", "qps3 = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.Y, Pauli.X])\n", "qps4 = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.X, Pauli.X])\n", "qps5 = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.X, Pauli.X])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now, create some symbolic expressions for the $a_j t_j$ terms."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import fresh_symbol"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["symbol1 = fresh_symbol(\"s0\")\n", "expr1 = 1.2 * symbol1\n", "symbol2 = fresh_symbol(\"s1\")\n", "expr2 = -0.3 * symbol2"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can now create our `QubitPauliOperator`."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils import QubitPauliOperator"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["dict1 = dict((string, expr1) for string in (qps0, qps1))\n", "dict2 = dict((string, expr2) for string in (qps2, qps3, qps4, qps5))\n", "operator = QubitPauliOperator({**dict1, **dict2})\n", "print(operator)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Now we can let `pytket` sequence the terms in this operator for us, using a selection of strategies. First, we will create a `Circuit` to generate an example reference state, and then use the `gen_term_sequence_circuit` method to append the Pauli exponentials."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import Circuit\n", "from pytket.utils import gen_term_sequence_circuit\n", "from pytket.partition import PauliPartitionStrat, GraphColourMethod"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["reference_circ = Circuit(4).X(1).X(3)\n", "ansatz_circuit = gen_term_sequence_circuit(\n", " operator, reference_circ, PauliPartitionStrat.CommutingSets, GraphColourMethod.Lazy\n", ")"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This method works by generating a graph of Pauli exponentials and performing graph colouring. Here we have chosen to partition the terms so that exponentials which commute are gathered together, and we have done so using a lazy, greedy graph colouring method.
\n", "
\n", "Alternatively, we could have used the `PauliPartitionStrat.NonConflictingSets`, which puts Pauli exponentials together so that they only require single-qubit gates to be converted into the form $e^{i \\alpha Z \\otimes Z \\otimes ... \\otimes Z}$. This strategy is primarily useful for measurement reduction, a different problem.
\n", "
\n", "We could also have used the `GraphColourMethod.LargestFirst`, which still uses a greedy method, but builds the full graph and iterates through the vertices in descending order of arity. We recommend playing around with the options, but we typically find that the combination of `CommutingSets` and `Lazy` allows the best optimisation.
\n", "
\n", "In general, not all of our exponentials will commute, so the semantics of our circuit depend on the order of our sequencing. As a result, it is important for us to be able to inspect the order we have produced. `pytket` provides functionality to enable this. Each set of commuting exponentials is put into a `CircBox`, which lets us inspect the partitoning."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import OpType"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["for command in ansatz_circuit:\n", " if command.op.type == OpType.CircBox:\n", " print(\"New CircBox:\")\n", " for pauli_exp in command.op.get_circuit():\n", " print(\n", " \" {} {} {}\".format(\n", " pauli_exp, pauli_exp.op.get_paulis(), pauli_exp.op.get_phase()\n", " )\n", " )\n", " else:\n", " print(\"Native gate: {}\".format(command))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can convert this circuit into basic gates using a `pytket` `Transform`. This acts in place on the circuit to do rewriting, for gate translation and optimisation. We will start off with a naive decomposition."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.transform import Transform"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["naive_circuit = ansatz_circuit.copy()\n", "Transform.DecomposeBoxes().apply(naive_circuit)\n", "print(naive_circuit.get_commands())"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is a jumble of one- and two-qubit gates. We can get some relevant circuit metrics out:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"Naive CX Depth: {}\".format(naive_circuit.depth_by_type(OpType.CX)))\n", "print(\"Naive CX Count: {}\".format(naive_circuit.n_gates_of_type(OpType.CX)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These metrics can be improved upon significantly by smart compilation. A `Transform` exists precisely for this purpose:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.transform import PauliSynthStrat, CXConfigType"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["smart_circuit = ansatz_circuit.copy()\n", "Transform.UCCSynthesis(PauliSynthStrat.Sets, CXConfigType.Tree).apply(smart_circuit)\n", "print(\"Smart CX Depth: {}\".format(smart_circuit.depth_by_type(OpType.CX)))\n", "print(\"Smart CX Count: {}\".format(smart_circuit.n_gates_of_type(OpType.CX)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This `Transform` takes in a `Circuit` with the structure specified above: some arbitrary gates for the reference state, along with several `CircBox` gates containing `PauliExpBox` gates.
\n", "
\n", "We have chosen `PauliSynthStrat.Sets` and `CXConfigType.Tree`. The `PauliSynthStrat` dictates the method for decomposing multiple adjacent Pauli exponentials into basic gates, while the `CXConfigType` dictates the structure of adjacent CX gates.
\n", "
\n", "If we choose a different combination of strategies, we can produce a different output circuit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["last_circuit = ansatz_circuit.copy()\n", "Transform.UCCSynthesis(PauliSynthStrat.Individual, CXConfigType.Snake).apply(\n", " last_circuit\n", ")\n", "print(last_circuit.get_commands())"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"Last CX Depth: {}\".format(last_circuit.depth_by_type(OpType.CX)))\n", "print(\"Last CX Count: {}\".format(last_circuit.n_gates_of_type(OpType.CX)))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Other than some single-qubit Cliffords we acquired via synthesis, you can check that this gives us the same circuit structure as our `Transform.DecomposeBoxes` method! It is a suboptimal synthesis method.
\n", "
\n", "As with the `gen_term_sequence` method, we recommend playing around with the arguments and seeing what circuits come out. Typically we find that `PauliSynthStrat.Sets` and `CXConfigType.Tree` work the best, although routing can affect this somewhat."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file diff --git a/examples/python/ansatz_sequence_example.py b/examples/python/ansatz_sequence_example.py index 35efcc50..b6b34b91 100644 --- a/examples/python/ansatz_sequence_example.py +++ b/examples/python/ansatz_sequence_example.py @@ -4,7 +4,7 @@ # # This notebook will describe how to use an advanced feature of `pytket` to enable automated circuit synthesis for $U$ and reduce circuit depth dramatically. # -# We must create a `pytket` `QubitPauliOperator`, which represents such an operator $U$, and contains a dictionary from Pauli string $P_{jk}$ to symbolic expression. Here, we make a mock operator ourselves, which resembles the UCCSD excitation operator for the $\mathrm{H}_2$ molecule using the Jordan-Wigner qubit encoding. In the future, operator generation will be handled automatically using CQC's upcoming software for enterprise quantum chemistry, EUMEN. We also offer conversion to and from the `OpenFermion` `QubitOperator` class, although at the time of writing a `QubitOperator` cannot handle arbitrary symbols. +# We must create a `pytket` `QubitPauliOperator`, which represents such an operator $U$, and contains a dictionary from Pauli string $P_{jk}$ to symbolic expression. Here, we make a mock operator ourselves, which resembles the UCCSD excitation operator for the $\mathrm{H}_2$ molecule using the Jordan-Wigner qubit encoding. # # First, we create a series of `QubitPauliString` objects, which represent each $P_{jk}$. From 7f7512c33a639cbcfefeadccb8a84aed272593e8 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Wed, 8 Nov 2023 01:18:11 +0000 Subject: [PATCH 42/51] fix links at the bottom of QPE notebook --- examples/phase_estimation.ipynb | 2 +- examples/python/phase_estimation.py | 9 ++------- 2 files changed, 3 insertions(+), 8 deletions(-) diff --git a/examples/phase_estimation.ipynb b/examples/phase_estimation.ipynb index f68b02f8..fd6b13e6 100644 --- a/examples/phase_estimation.ipynb +++ b/examples/phase_estimation.ipynb @@ -1 +1 @@ -{"cells":[{"cell_type":"markdown","metadata":{},"source":["# Quantum Phase Estimation\n","\n","When constructing circuits for quantum algorithms it is useful to think of higher level operations than just individual quantum gates.\n","\n","In `pytket` we can construct circuits using box structures which abstract away the complexity of the underlying circuit.\n","\n","This notebook is intended to complement the [boxes section](https://tket.quantinuum.com/user-manual/manual_circuit.html#boxes) of the user manual which introduces the different box types.\n","\n","To demonstrate boxes in `pytket` we will consider the Quantum Phase Estimation algorithm (QPE). This is an important subroutine in several quantum algorithms including Shor's algorithm and fault-tolerant approaches to quantum chemistry.\n","\n","## Overview of Phase Estimation\n","\n","The Quantum Phase Estimation algorithm can be used to estimate the eigenvalues of some unitary operator $U$ to some desired precision.\n","\n","The eigenvalues of $U$ lie on the unit circle, giving us the following eigenvalue equation\n","\n","$$\n","\\begin{equation}\n","U |\\psi \\rangle = e^{2 \\pi i \\theta} |\\psi\\rangle\\,, \\quad 0 \\leq \\theta \\leq 1\n","\\end{equation}\n","$$\n","\n","Here $|\\psi \\rangle$ is an eigenstate of the operator $U$. In phase estimation we estimate the eigenvalue $e^{2 \\pi i \\theta}$ by approximating $\\theta$.\n","\n","\n","The circuit for Quantum phase estimation is itself composed of several subroutines which we can realise as boxes.\n","\n","![](images/phase_est.png \"Quantum Phase Estimation Circuit\")"]},{"cell_type":"markdown","metadata":{},"source":["QPE is generally split up into three stages\n","\n","1. Firstly we prepare an initial state in one register. In parallel we prepare a uniform superposition state using Hadamard gates on some ancilla qubits. The number of ancilla qubits determines how precisely we can estimate the phase $\\theta$.\n","\n","2. Secondly we apply successive controlled $U$ gates. This has the effect of \"kicking back\" phases onto the ancilla qubits according to the eigenvalue equation above.\n","\n","3. Finally we apply the inverse Quantum Fourier Transform (QFT). This essentially plays the role of destructive interference, suppressing amplitudes from \"undesirable states\" and hopefully allowing us to measure a single outcome (or a small number of outcomes) with high probability.\n","\n","\n","There is some subtlety around the first point. The initial state used can be an exact eigenstate of $U$ however this may be difficult to prepare if we don't know the eigenvalues of $U$ in advance. Alternatively we could use an initial state that is a linear combination of eigenstates, as the phase estimation will project into the eigenspace of $U$."]},{"cell_type":"markdown","metadata":{},"source":["We also assume that we can implement $U$ with a quantum circuit. In chemistry applications $U$ could be of the form $U=e^{-iHt}$ where $H$ is the Hamiltonian of some system of interest. In the cannonical algorithm, the number of controlled unitaries we apply scales exponentially with the number of ancilla qubits. This allows more precision at the expense of a larger quantum circuit."]},{"cell_type":"markdown","metadata":{},"source":["## The Quantum Fourier Transform"]},{"cell_type":"markdown","metadata":{},"source":["Before considering the other parts of the QPE algorithm, lets focus on the Quantum Fourier Transform (QFT) subroutine.\n","\n","Mathematically, the QFT has the following action.\n","\n","$$\n","\\begin{equation}\n","QFT : |j\\rangle\\ \\longmapsto \\sum_{k=0}^{N - 1} e^{2 \\pi ijk/N}|k\\rangle, \\quad N= 2^k\n","\\end{equation}\n","$$\n","\n","This is essentially the Discrete Fourier transform except the input is a quantum state $|j\\rangle$.\n","\n","It is well known that the QFT can be implemented efficiently with a quantum circuit\n","\n","We can build the circuit for the $n$ qubit QFT using $n$ Hadamard gates $\\frac{n}{2}$ swap gates and $\\frac{n(n-1)}{2}$ controlled unitary rotations $\\text{CU1}$.\n","\n","$$\n"," \\begin{equation}\n"," CU1(\\phi) =\n"," \\begin{pmatrix}\n"," I & 0 \\\\\n"," 0 & U1(\\phi)\n"," \\end{pmatrix}\n"," \\,, \\quad\n","U1(\\phi) =\n"," \\begin{pmatrix}\n"," 1 & 0 \\\\\n"," 0 & e^{i \\phi}\n"," \\end{pmatrix}\n"," \\end{equation}\n","$$\n","\n","The circuit for the Quantum Fourier transform on three qubits is the following\n","\n","![](images/qft.png \"QFT Circuit\")\n","\n","We can build this circuit in `pytket` by adding gate operations manually:"]},{"cell_type":"markdown","metadata":{},"source":["lets build the QFT for three qubits"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import Circuit\n","from pytket.circuit.display import render_circuit_jupyter"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft3_circ = Circuit(3)\n","qft3_circ.H(0)\n","qft3_circ.CU1(0.5, 1, 0)\n","qft3_circ.CU1(0.25, 2, 0)\n","qft3_circ.H(1)\n","qft3_circ.CU1(0.5, 2, 1)\n","qft3_circ.H(2)\n","qft3_circ.SWAP(0, 2)\n","render_circuit_jupyter(qft3_circ)"]},{"cell_type":"markdown","metadata":{},"source":["We can generalise the quantum Fourier transform to $n$ qubits by iterating over the qubits as follows"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def build_qft_circuit(n_qubits: int) -> Circuit:\n"," circ = Circuit(n_qubits, name=\"QFT\")\n"," for i in range(n_qubits):\n"," circ.H(i)\n"," for j in range(i + 1, n_qubits):\n"," circ.CU1(1 / 2 ** (j - i), j, i)\n"," for k in range(0, n_qubits // 2):\n"," circ.SWAP(k, n_qubits - k - 1)\n"," return circ"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft4_circ: Circuit = build_qft_circuit(4)\n","render_circuit_jupyter(qft4_circ)"]},{"cell_type":"markdown","metadata":{},"source":["Now that we have the generalised circuit we can wrap it up in a `CircBox` which can then be added to another circuit as a subroutine."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import CircBox"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft4_box: CircBox = CircBox(qft4_circ)\n","qft_circ = Circuit(4).add_gate(qft4_box, [0, 1, 2, 3])\n","render_circuit_jupyter(qft_circ)"]},{"cell_type":"markdown","metadata":{},"source":["Note how the `CircBox` inherits the name `QFT` from the underlying circuit."]},{"cell_type":"markdown","metadata":{},"source":["Recall that in our phase estimation algorithm we need to use the inverse QFT.\n","\n","$$\n","\\begin{equation}\n","\\text{QFT}^† : \\sum_{k=0}^{N - 1} e^{2 \\pi ijk/N}|k\\rangle \\longmapsto |j\\rangle\\,, \\quad N= 2^k\n","\\end{equation}\n","$$\n","\n","\n","Now that we have the QFT circuit we can obtain the inverse by using `CircBox.dagger`. We can also verify that this is correct by inspecting the circuit inside with `CircBox.get_circuit()`."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["inv_qft4_box = qft4_box.dagger\n","render_circuit_jupyter(inv_qft4_box.get_circuit())"]},{"cell_type":"markdown","metadata":{},"source":["## The Controlled Unitary Operations"]},{"cell_type":"markdown","metadata":{},"source":["In the phase estimation algorithm we repeatedly perform controlled unitary operations. In the canonical variant, the number of controlled unitaries will be $2^m - 1$ where $m$ is the number of measurement qubits."]},{"cell_type":"markdown","metadata":{},"source":["The form of $U$ will vary depending on the application. For chemistry or condensed matter physics $U$ typically be the time evolution operator $U(t) = e^{- i H t}$ where $H$ is the problem Hamiltonian."]},{"cell_type":"markdown","metadata":{},"source":["Suppose that we had the following decomposition for $H$ in terms of Pauli strings $P_j$ and complex coefficients $\\alpha_j$.\n","\n","$$\n","\\begin{equation}\n","H = \\sum_j \\alpha_j P_j\\,, \\quad \\, P_j \\in \\{I, \\,X, \\,Y, \\,Z\\}^{\\otimes n}\n","\\end{equation}\n","$$\n","\n","Here Pauli strings refers to tensor products of Pauli operators. These strings form an orthonormal basis for $2^n \\times 2^n$ matrices."]},{"cell_type":"markdown","metadata":{},"source":["If we have a Hamiltonian in the form above, we can then implement $U(t)$ as a sequence of Pauli gadget circuits. We can do this with the [PauliExpBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.PauliExpBox) construct in pytket. For more on `PauliExpBox` see the [user manual](https://tket.quantinuum.com/user-manual/manual_circuit.html#pauli-exponential-boxes)."]},{"cell_type":"markdown","metadata":{},"source":["Once we have a circuit to implement our time evolution operator $U(t)$, we can construct the controlled $U(t)$ operations using [QControlBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.QControlBox). If our base unitary is a sequence of `PauliExpBox`(es) then there is some structure we can exploit to simplify our circuit. See this [blog post](https://tket.quantinuum.com/tket-blog/posts/controlled_gates/) on [ConjugationBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.ConjugationBox) for more."]},{"cell_type":"markdown","metadata":{},"source":["In what follows, we will just construct a simplified instance of QPE where the controlled unitaries are just $\\text{CU1}$ gates."]},{"cell_type":"markdown","metadata":{},"source":["## Putting it all together"]},{"cell_type":"markdown","metadata":{},"source":["We can now define a function to build our entire QPE circuit. We can make this function take a state preparation circuit and a unitary circuit as input as well. The function also has the number of measurement qubits as input which will determine the precision of our phase estimate."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import QControlBox"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def build_phase_est_circuit(\n"," n_measurement_qubits: int, state_prep_circuit: Circuit, unitary_circuit: Circuit\n",") -> Circuit:\n"," qpe_circ: Circuit = Circuit()\n"," n_state_prep_qubits = state_prep_circuit.n_qubits\n"," measurement_register = qpe_circ.add_q_register(\"m\", n_measurement_qubits)\n"," state_prep_register = qpe_circ.add_q_register(\"p\", n_state_prep_qubits)\n"," qpe_circ.add_circuit(state_prep_circuit, list(state_prep_register))\n","\n"," # Create a controlled unitary with a single control qubit\n"," unitary_circuit.name = \"U\"\n"," controlled_u_gate = QControlBox(CircBox(unitary_circuit), 1)\n","\n"," # Add Hadamard gates to every qubit in the measurement register\n"," for m_qubit in measurement_register:\n"," qpe_circ.H(m_qubit)\n","\n"," # Add all (2**n_measurement_qubits - 1) of the controlled unitaries sequentially\n"," for m_qubit in range(n_measurement_qubits):\n"," control_index = n_measurement_qubits - m_qubit - 1\n"," control_qubit = [measurement_register[control_index]]\n"," for _ in range(2**m_qubit):\n"," qpe_circ.add_qcontrolbox(\n"," controlled_u_gate, control_qubit + list(state_prep_register)\n"," )\n","\n"," # Finally, append the inverse qft and measure the qubits\n"," qft_box = CircBox(build_qft_circuit(n_measurement_qubits))\n"," inverse_qft_box = qft_box.dagger\n"," qpe_circ.add_circbox(inverse_qft_box, list(measurement_register))\n"," qpe_circ.measure_register(measurement_register, \"c\")\n"," return qpe_circ"]},{"cell_type":"markdown","metadata":{},"source":["## Phase Estimation with a Trivial Eigenstate\n","\n","Lets test our circuit construction by preparing a trivial $|1\\rangle$ eigenstate of the $\\text{U1}$ gate. We can then see if our phase estimation circuit returns the expected eigenvalue."]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","U1(\\phi)|1\\rangle = e^{i\\phi} = e^{2 \\pi i \\theta} \\implies \\theta = \\frac{\\phi}{2}\n","\\end{equation}\n","$$\n","\n","So we expect that our ideal phase $\\theta$ will be half the input angle $\\phi$ to our $U1$ gate."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["prep_circuit = Circuit(1).X(0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["input_angle = 0.73 # angle as number of half turns"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["unitary_circuit = Circuit(1).U1(input_angle, 0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qpe_circ_trivial = build_phase_est_circuit(\n"," 4, state_prep_circuit=prep_circuit, unitary_circuit=unitary_circuit\n",")"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["render_circuit_jupyter(qpe_circ_trivial)"]},{"cell_type":"markdown","metadata":{},"source":["Lets use the noiseless `AerBackend` simulator to run our phase estimation circuit."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.extensions.qiskit import AerBackend"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["backend = AerBackend()"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["compiled_circ = backend.get_compiled_circuit(qpe_circ_trivial)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_shots = 1000\n","result = backend.run_circuit(compiled_circ, n_shots)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(result.get_counts())"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.backends.backendresult import BackendResult\n","import matplotlib.pyplot as plt"]},{"cell_type":"markdown","metadata":{},"source":["plotting function for QPE Notebook"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def plot_qpe_results(\n"," sim_result: BackendResult,\n"," n_strings: int = 4,\n"," dark_mode: bool = False,\n"," y_limit: int = 1000,\n",") -> None:\n"," \"\"\"\n"," Plots results in a barchart given a BackendResult. the number of stings displayed\n"," can be specified with the n_strings argument.\n"," \"\"\"\n"," counts_dict = sim_result.get_counts()\n"," sorted_shots = counts_dict.most_common()\n"," n_most_common_strings = sorted_shots[:n_strings]\n"," x_axis_values = [str(entry[0]) for entry in n_most_common_strings] # basis states\n"," y_axis_values = [entry[1] for entry in n_most_common_strings] # counts\n"," if dark_mode:\n"," plt.style.use(\"dark_background\")\n"," fig = plt.figure()\n"," ax = fig.add_axes((0, 0, 0.75, 0.5))\n"," color_list = [\"orange\"] * (len(x_axis_values))\n"," ax.bar(\n"," x=x_axis_values,\n"," height=y_axis_values,\n"," color=color_list,\n"," )\n"," ax.set_title(label=\"Results\")\n"," plt.ylim([0, y_limit])\n"," plt.xlabel(\"Basis State\")\n"," plt.ylabel(\"Number of Shots\")\n"," plt.show()"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plot_qpe_results(result, y_limit=int(1.2 * n_shots))"]},{"cell_type":"markdown","metadata":{},"source":["As expected we see one outcome with high probability. Lets now extract our approximation of $\\theta$ from our output bitstrings.\n","\n","suppose the $j$ is an integer representation of our most commonly measured bitstring."]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","\\theta_{estimate} = \\frac{j}{N}\n","\\end{equation}\n","$$"]},{"cell_type":"markdown","metadata":{},"source":["Here $N = 2 ^n$ where $n$ is the number of measurement qubits."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.backends.backendresult import BackendResult"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def single_phase_from_backendresult(result: BackendResult) -> float:\n"," # Extract most common measurement outcome\n"," basis_state = result.get_counts().most_common()[0][0]\n"," bitstring = \"\".join([str(bit) for bit in basis_state])\n"," integer = int(bitstring, 2)\n","\n"," # Calculate theta estimate\n"," return integer / (2 ** len(bitstring))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["theta = single_phase_from_backendresult(result)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(theta)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(input_angle / 2)"]},{"cell_type":"markdown","metadata":{},"source":["Our output is close to half our input angle $\\phi$ as expected. Lets calculate our error."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["error = round(abs(input_angle - (2 * theta)), 3)\n","print(error)"]},{"cell_type":"markdown","metadata":{},"source":["## Suggestions for further reading\n","\n","In this notebook we have shown the canonical variant of quantum phase estimation. There are several other variants.\n","\n","Quantinuum paper on Bayesian phase estimation -> https://arxiv.org/pdf/2306.16608.pdf\n","Blog post on `ConjugationBox` -> https://tket.quantinuum.com/tket-blog/posts/controlled_gates/ - efficient circuits for controlled Pauli gadgets.\n","\n","As mentioned quantum phase estimation is a subroutine in Shor's algorithm. Read more about how phase estimation is used in period finding."]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} +{"cells":[{"cell_type":"markdown","metadata":{},"source":["# Quantum Phase Estimation\n","\n","When constructing circuits for quantum algorithms it is useful to think of higher level operations than just individual quantum gates.\n","\n","In `pytket` we can construct circuits using box structures which abstract away the complexity of the underlying circuit.\n","\n","This notebook is intended to complement the [boxes section](https://tket.quantinuum.com/user-manual/manual_circuit.html#boxes) of the user manual which introduces the different box types.\n","\n","To demonstrate boxes in `pytket` we will consider the Quantum Phase Estimation algorithm (QPE). This is an important subroutine in several quantum algorithms including Shor's algorithm and fault-tolerant approaches to quantum chemistry.\n","\n","## Overview of Phase Estimation\n","\n","The Quantum Phase Estimation algorithm can be used to estimate the eigenvalues of some unitary operator $U$ to some desired precision.\n","\n","The eigenvalues of $U$ lie on the unit circle, giving us the following eigenvalue equation\n","\n","$$\n","\\begin{equation}\n","U |\\psi \\rangle = e^{2 \\pi i \\theta} |\\psi\\rangle\\,, \\quad 0 \\leq \\theta \\leq 1\n","\\end{equation}\n","$$\n","\n","Here $|\\psi \\rangle$ is an eigenstate of the operator $U$. In phase estimation we estimate the eigenvalue $e^{2 \\pi i \\theta}$ by approximating $\\theta$.\n","\n","\n","The circuit for Quantum phase estimation is itself composed of several subroutines which we can realise as boxes.\n","\n","![](images/phase_est.png \"Quantum Phase Estimation Circuit\")"]},{"cell_type":"markdown","metadata":{},"source":["QPE is generally split up into three stages\n","\n","1. Firstly we prepare an initial state in one register. In parallel we prepare a uniform superposition state using Hadamard gates on some ancilla qubits. The number of ancilla qubits determines how precisely we can estimate the phase $\\theta$.\n","\n","2. Secondly we apply successive controlled $U$ gates. This has the effect of \"kicking back\" phases onto the ancilla qubits according to the eigenvalue equation above.\n","\n","3. Finally we apply the inverse Quantum Fourier Transform (QFT). This essentially plays the role of destructive interference, suppressing amplitudes from \"undesirable states\" and hopefully allowing us to measure a single outcome (or a small number of outcomes) with high probability.\n","\n","\n","There is some subtlety around the first point. The initial state used can be an exact eigenstate of $U$ however this may be difficult to prepare if we don't know the eigenvalues of $U$ in advance. Alternatively we could use an initial state that is a linear combination of eigenstates, as the phase estimation will project into the eigenspace of $U$."]},{"cell_type":"markdown","metadata":{},"source":["We also assume that we can implement $U$ with a quantum circuit. In chemistry applications $U$ could be of the form $U=e^{-iHt}$ where $H$ is the Hamiltonian of some system of interest. In the cannonical algorithm, the number of controlled unitaries we apply scales exponentially with the number of ancilla qubits. This allows more precision at the expense of a larger quantum circuit."]},{"cell_type":"markdown","metadata":{},"source":["## The Quantum Fourier Transform"]},{"cell_type":"markdown","metadata":{},"source":["Before considering the other parts of the QPE algorithm, lets focus on the Quantum Fourier Transform (QFT) subroutine.\n","\n","Mathematically, the QFT has the following action.\n","\n","$$\n","\\begin{equation}\n","QFT : |j\\rangle\\ \\longmapsto \\sum_{k=0}^{N - 1} e^{2 \\pi ijk/N}|k\\rangle, \\quad N= 2^k\n","\\end{equation}\n","$$\n","\n","This is essentially the Discrete Fourier transform except the input is a quantum state $|j\\rangle$.\n","\n","It is well known that the QFT can be implemented efficiently with a quantum circuit\n","\n","We can build the circuit for the $n$ qubit QFT using $n$ Hadamard gates $\\frac{n}{2}$ swap gates and $\\frac{n(n-1)}{2}$ controlled unitary rotations $\\text{CU1}$.\n","\n","$$\n"," \\begin{equation}\n"," CU1(\\phi) =\n"," \\begin{pmatrix}\n"," I & 0 \\\\\n"," 0 & U1(\\phi)\n"," \\end{pmatrix}\n"," \\,, \\quad\n","U1(\\phi) =\n"," \\begin{pmatrix}\n"," 1 & 0 \\\\\n"," 0 & e^{i \\phi}\n"," \\end{pmatrix}\n"," \\end{equation}\n","$$\n","\n","The circuit for the Quantum Fourier transform on three qubits is the following\n","\n","![](images/qft.png \"QFT Circuit\")\n","\n","We can build this circuit in `pytket` by adding gate operations manually:"]},{"cell_type":"markdown","metadata":{},"source":["lets build the QFT for three qubits"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import Circuit\n","from pytket.circuit.display import render_circuit_jupyter"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft3_circ = Circuit(3)\n","qft3_circ.H(0)\n","qft3_circ.CU1(0.5, 1, 0)\n","qft3_circ.CU1(0.25, 2, 0)\n","qft3_circ.H(1)\n","qft3_circ.CU1(0.5, 2, 1)\n","qft3_circ.H(2)\n","qft3_circ.SWAP(0, 2)\n","render_circuit_jupyter(qft3_circ)"]},{"cell_type":"markdown","metadata":{},"source":["We can generalise the quantum Fourier transform to $n$ qubits by iterating over the qubits as follows"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def build_qft_circuit(n_qubits: int) -> Circuit:\n"," circ = Circuit(n_qubits, name=\"QFT\")\n"," for i in range(n_qubits):\n"," circ.H(i)\n"," for j in range(i + 1, n_qubits):\n"," circ.CU1(1 / 2 ** (j - i), j, i)\n"," for k in range(0, n_qubits // 2):\n"," circ.SWAP(k, n_qubits - k - 1)\n"," return circ"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft4_circ: Circuit = build_qft_circuit(4)\n","render_circuit_jupyter(qft4_circ)"]},{"cell_type":"markdown","metadata":{},"source":["Now that we have the generalised circuit we can wrap it up in a `CircBox` which can then be added to another circuit as a subroutine."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import CircBox"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qft4_box: CircBox = CircBox(qft4_circ)\n","qft_circ = Circuit(4).add_gate(qft4_box, [0, 1, 2, 3])\n","render_circuit_jupyter(qft_circ)"]},{"cell_type":"markdown","metadata":{},"source":["Note how the `CircBox` inherits the name `QFT` from the underlying circuit."]},{"cell_type":"markdown","metadata":{},"source":["Recall that in our phase estimation algorithm we need to use the inverse QFT.\n","\n","$$\n","\\begin{equation}\n","\\text{QFT}^† : \\sum_{k=0}^{N - 1} e^{2 \\pi ijk/N}|k\\rangle \\longmapsto |j\\rangle\\,, \\quad N= 2^k\n","\\end{equation}\n","$$\n","\n","\n","Now that we have the QFT circuit we can obtain the inverse by using `CircBox.dagger`. We can also verify that this is correct by inspecting the circuit inside with `CircBox.get_circuit()`."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["inv_qft4_box = qft4_box.dagger\n","render_circuit_jupyter(inv_qft4_box.get_circuit())"]},{"cell_type":"markdown","metadata":{},"source":["## The Controlled Unitary Operations"]},{"cell_type":"markdown","metadata":{},"source":["In the phase estimation algorithm we repeatedly perform controlled unitary operations. In the canonical variant, the number of controlled unitaries will be $2^m - 1$ where $m$ is the number of measurement qubits."]},{"cell_type":"markdown","metadata":{},"source":["The form of $U$ will vary depending on the application. For chemistry or condensed matter physics $U$ typically be the time evolution operator $U(t) = e^{- i H t}$ where $H$ is the problem Hamiltonian."]},{"cell_type":"markdown","metadata":{},"source":["Suppose that we had the following decomposition for $H$ in terms of Pauli strings $P_j$ and complex coefficients $\\alpha_j$.\n","\n","$$\n","\\begin{equation}\n","H = \\sum_j \\alpha_j P_j\\,, \\quad \\, P_j \\in \\{I, \\,X, \\,Y, \\,Z\\}^{\\otimes n}\n","\\end{equation}\n","$$\n","\n","Here Pauli strings refers to tensor products of Pauli operators. These strings form an orthonormal basis for $2^n \\times 2^n$ matrices."]},{"cell_type":"markdown","metadata":{},"source":["If we have a Hamiltonian in the form above, we can then implement $U(t)$ as a sequence of Pauli gadget circuits. We can do this with the [PauliExpBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.PauliExpBox) construct in pytket. For more on `PauliExpBox` see the [user manual](https://tket.quantinuum.com/user-manual/manual_circuit.html#pauli-exponential-boxes)."]},{"cell_type":"markdown","metadata":{},"source":["Once we have a circuit to implement our time evolution operator $U(t)$, we can construct the controlled $U(t)$ operations using [QControlBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.QControlBox). If our base unitary is a sequence of `PauliExpBox`(es) then there is some structure we can exploit to simplify our circuit. See this [blog post](https://tket.quantinuum.com/tket-blog/posts/controlled_gates/) on [ConjugationBox](https://tket.quantinuum.com/api-docs/circuit.html#pytket.circuit.ConjugationBox) for more."]},{"cell_type":"markdown","metadata":{},"source":["In what follows, we will just construct a simplified instance of QPE where the controlled unitaries are just $\\text{CU1}$ gates."]},{"cell_type":"markdown","metadata":{},"source":["## Putting it all together"]},{"cell_type":"markdown","metadata":{},"source":["We can now define a function to build our entire QPE circuit. We can make this function take a state preparation circuit and a unitary circuit as input as well. The function also has the number of measurement qubits as input which will determine the precision of our phase estimate."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.circuit import QControlBox"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def build_phase_est_circuit(\n"," n_measurement_qubits: int, state_prep_circuit: Circuit, unitary_circuit: Circuit\n",") -> Circuit:\n"," qpe_circ: Circuit = Circuit()\n"," n_state_prep_qubits = state_prep_circuit.n_qubits\n"," measurement_register = qpe_circ.add_q_register(\"m\", n_measurement_qubits)\n"," state_prep_register = qpe_circ.add_q_register(\"p\", n_state_prep_qubits)\n"," qpe_circ.add_circuit(state_prep_circuit, list(state_prep_register))\n","\n"," # Create a controlled unitary with a single control qubit\n"," unitary_circuit.name = \"U\"\n"," controlled_u_gate = QControlBox(CircBox(unitary_circuit), 1)\n","\n"," # Add Hadamard gates to every qubit in the measurement register\n"," for m_qubit in measurement_register:\n"," qpe_circ.H(m_qubit)\n","\n"," # Add all (2**n_measurement_qubits - 1) of the controlled unitaries sequentially\n"," for m_qubit in range(n_measurement_qubits):\n"," control_index = n_measurement_qubits - m_qubit - 1\n"," control_qubit = [measurement_register[control_index]]\n"," for _ in range(2**m_qubit):\n"," qpe_circ.add_qcontrolbox(\n"," controlled_u_gate, control_qubit + list(state_prep_register)\n"," )\n","\n"," # Finally, append the inverse qft and measure the qubits\n"," qft_box = CircBox(build_qft_circuit(n_measurement_qubits))\n"," inverse_qft_box = qft_box.dagger\n"," qpe_circ.add_circbox(inverse_qft_box, list(measurement_register))\n"," qpe_circ.measure_register(measurement_register, \"c\")\n"," return qpe_circ"]},{"cell_type":"markdown","metadata":{},"source":["## Phase Estimation with a Trivial Eigenstate\n","\n","Lets test our circuit construction by preparing a trivial $|1\\rangle$ eigenstate of the $\\text{U1}$ gate. We can then see if our phase estimation circuit returns the expected eigenvalue."]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","U1(\\phi)|1\\rangle = e^{i\\phi} = e^{2 \\pi i \\theta} \\implies \\theta = \\frac{\\phi}{2}\n","\\end{equation}\n","$$\n","\n","So we expect that our ideal phase $\\theta$ will be half the input angle $\\phi$ to our $U1$ gate."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["prep_circuit = Circuit(1).X(0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["input_angle = 0.73 # angle as number of half turns"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["unitary_circuit = Circuit(1).U1(input_angle, 0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qpe_circ_trivial = build_phase_est_circuit(\n"," 4, state_prep_circuit=prep_circuit, unitary_circuit=unitary_circuit\n",")"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["render_circuit_jupyter(qpe_circ_trivial)"]},{"cell_type":"markdown","metadata":{},"source":["Lets use the noiseless `AerBackend` simulator to run our phase estimation circuit."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.extensions.qiskit import AerBackend"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["backend = AerBackend()"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["compiled_circ = backend.get_compiled_circuit(qpe_circ_trivial)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_shots = 1000\n","result = backend.run_circuit(compiled_circ, n_shots)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(result.get_counts())"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.backends.backendresult import BackendResult\n","import matplotlib.pyplot as plt"]},{"cell_type":"markdown","metadata":{},"source":["plotting function for QPE Notebook"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def plot_qpe_results(\n"," sim_result: BackendResult,\n"," n_strings: int = 4,\n"," dark_mode: bool = False,\n"," y_limit: int = 1000,\n",") -> None:\n"," \"\"\"\n"," Plots results in a barchart given a BackendResult. the number of stings displayed\n"," can be specified with the n_strings argument.\n"," \"\"\"\n"," counts_dict = sim_result.get_counts()\n"," sorted_shots = counts_dict.most_common()\n"," n_most_common_strings = sorted_shots[:n_strings]\n"," x_axis_values = [str(entry[0]) for entry in n_most_common_strings] # basis states\n"," y_axis_values = [entry[1] for entry in n_most_common_strings] # counts\n"," if dark_mode:\n"," plt.style.use(\"dark_background\")\n"," fig = plt.figure()\n"," ax = fig.add_axes((0, 0, 0.75, 0.5))\n"," color_list = [\"orange\"] * (len(x_axis_values))\n"," ax.bar(\n"," x=x_axis_values,\n"," height=y_axis_values,\n"," color=color_list,\n"," )\n"," ax.set_title(label=\"Results\")\n"," plt.ylim([0, y_limit])\n"," plt.xlabel(\"Basis State\")\n"," plt.ylabel(\"Number of Shots\")\n"," plt.show()"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plot_qpe_results(result, y_limit=int(1.2 * n_shots))"]},{"cell_type":"markdown","metadata":{},"source":["As expected we see one outcome with high probability. Lets now extract our approximation of $\\theta$ from our output bitstrings.\n","\n","suppose the $j$ is an integer representation of our most commonly measured bitstring."]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","\\theta_{estimate} = \\frac{j}{N}\n","\\end{equation}\n","$$"]},{"cell_type":"markdown","metadata":{},"source":["Here $N = 2 ^n$ where $n$ is the number of measurement qubits."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.backends.backendresult import BackendResult"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def single_phase_from_backendresult(result: BackendResult) -> float:\n"," # Extract most common measurement outcome\n"," basis_state = result.get_counts().most_common()[0][0]\n"," bitstring = \"\".join([str(bit) for bit in basis_state])\n"," integer = int(bitstring, 2)\n","\n"," # Calculate theta estimate\n"," return integer / (2 ** len(bitstring))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["theta = single_phase_from_backendresult(result)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(theta)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(input_angle / 2)"]},{"cell_type":"markdown","metadata":{},"source":["Our output is close to half our input angle $\\phi$ as expected. Lets calculate our error."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["error = round(abs(input_angle - (2 * theta)), 3)\n","print(error)"]},{"cell_type":"markdown","metadata":{},"source":["## Suggestions for further reading\n","\n","* Quantinuum paper on Bayesian phase estimation -> https://arxiv.org/pdf/2306.16608.pdf\n","* Blog post on `ConjugationBox` (efficient circuits for controlled gates) -> https://tket.quantinuum.com/blog/posts/controlled_gates/"]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} diff --git a/examples/python/phase_estimation.py b/examples/python/phase_estimation.py index dcafd83c..7799a133 100644 --- a/examples/python/phase_estimation.py +++ b/examples/python/phase_estimation.py @@ -325,7 +325,6 @@ def single_phase_from_backendresult(result: BackendResult) -> float: theta = single_phase_from_backendresult(result) - print(theta) @@ -341,9 +340,5 @@ def single_phase_from_backendresult(result: BackendResult) -> float: # ## Suggestions for further reading # -# In this notebook we have shown the canonical variant of quantum phase estimation. There are several other variants. -# -# Quantinuum paper on Bayesian phase estimation -> https://arxiv.org/pdf/2306.16608.pdf -# Blog post on `ConjugationBox` -> https://tket.quantinuum.com/tket-blog/posts/controlled_gates/ - efficient circuits for controlled Pauli gadgets. -# -# As mentioned quantum phase estimation is a subroutine in Shor's algorithm. Read more about how phase estimation is used in period finding. +# * Quantinuum paper on Bayesian phase estimation -> https://arxiv.org/pdf/2306.16608.pdf +# * Blog post on `ConjugationBox` (efficient circuits for controlled gates) -> https://tket.quantinuum.com/blog/posts/controlled_gates/ From 34ed450574cb8b9a24ec706535ccfdf18a699a08 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Wed, 8 Nov 2023 01:20:23 +0000 Subject: [PATCH 43/51] delete unused Hamiltonian JSON --- examples/h2_5A.json | 409 -------------------------------------------- 1 file changed, 409 deletions(-) delete mode 100644 examples/h2_5A.json diff --git a/examples/h2_5A.json b/examples/h2_5A.json deleted file mode 100644 index a5f829c0..00000000 --- a/examples/h2_5A.json +++ /dev/null @@ -1,409 +0,0 @@ -[ - { - "string": [], - "coefficient": [ - -0.5458607027942332, - 0.0 - ] - }, - { - "string": [ - [ - [ - "q", - [ - 0 - ] - ], - "Z" - ] - ], - "coefficient": [ - 0.03970104575308908, - 0.0 - ] - }, - { - "string": [ - [ - [ - "q", - [ - 2 - ] - ], - "Z" - ] - ], - "coefficient": [ - 0.039577818133605905, - 0.0 - ] - }, - { - "string": [ - [ - [ - "q", - [ - 1 - ] - ], - "Z" - ] - ], - "coefficient": [ - 0.03970104575308908, - 0.0 - ] - }, - { - "string": [ - [ - [ - "q", - [ - 3 - ] - ], - "Z" - ] - ], - "coefficient": [ - 0.039577818133605905, - 0.0 - ] - }, - { - "string": [ - [ - [ - "q", - [ - 0 - ] - ], - "Z" - ], - [ - [ - "q", - [ - 2 - ] - ], - "Z" - ] - ], - "coefficient": [ - 0.026458859479781598, - 0.0 - ] - }, - { - "string": [ - [ - [ - "q", - [ - 1 - ] - ], - "Z" - ], - [ - [ - "q", - [ - 3 - ] - ], - "Z" - ] - ], - "coefficient": [ - 0.026458859479781598, - 0.0 - ] - }, - { - "string": [ - [ - [ - "q", - [ - 0 - ] - ], - "Z" - ], - [ - [ - "q", - [ - 1 - ] - ], - "Z" - ] - ], - "coefficient": [ - 0.11004294256569602, - 0.0 - ] - }, - { - "string": [ - [ - [ - "q", - [ - 1 - ] - ], - "Z" - ], - [ - [ - "q", - [ - 2 - ] - ], - "Z" - ] - ], - "coefficient": [ - 0.11005517318966757, - 0.0 - ] - }, - { - "string": [ - [ - [ - "q", - [ - 0 - ] - ], - "Y" - ], - [ - [ - "q", - [ - 1 - ] - ], - "X" - ], - [ - [ - "q", - [ - 2 - ] - ], - "X" - ], - [ - [ - "q", - [ - 3 - ] - ], - "Y" - ] - ], - "coefficient": [ - 0.08359631370988596, - 0.0 - ] - }, - { - "string": [ - [ - [ - "q", - [ - 0 - ] - ], - "X" - ], - [ - [ - "q", - [ - 1 - ] - ], - "X" - ], - [ - [ - "q", - [ - 2 - ] - ], - "Y" - ], - [ - [ - "q", - [ - 3 - ] - ], - "Y" - ] - ], - "coefficient": [ - -0.08359631370988596, - 0.0 - ] - }, - { - "string": [ - [ - [ - "q", - [ - 0 - ] - ], - "Y" - ], - [ - [ - "q", - [ - 1 - ] - ], - "Y" - ], - [ - [ - "q", - [ - 2 - ] - ], - "X" - ], - [ - [ - "q", - [ - 3 - ] - ], - "X" - ] - ], - "coefficient": [ - -0.08359631370988596, - 0.0 - ] - }, - { - "string": [ - [ - [ - "q", - [ - 0 - ] - ], - "X" - ], - [ - [ - "q", - [ - 1 - ] - ], - "Y" - ], - [ - [ - "q", - [ - 2 - ] - ], - "Y" - ], - [ - [ - "q", - [ - 3 - ] - ], - "X" - ] - ], - "coefficient": [ - 0.08359631370988596, - 0.0 - ] - }, - { - "string": [ - [ - [ - "q", - [ - 0 - ] - ], - "Z" - ], - [ - [ - "q", - [ - 3 - ] - ], - "Z" - ] - ], - "coefficient": [ - 0.11005517318966757, - 0.0 - ] - }, - { - "string": [ - [ - [ - "q", - [ - 2 - ] - ], - "Z" - ], - [ - [ - "q", - [ - 3 - ] - ], - "Z" - ] - ], - "coefficient": [ - 0.11006740930024865, - 0.0 - ] - } -] \ No newline at end of file From 801d8f00b34b361004dd44c34945410bb522a65a Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Wed, 8 Nov 2023 01:22:02 +0000 Subject: [PATCH 44/51] delete benchmarking example from config file --- examples/_config.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/examples/_config.yml b/examples/_config.yml index b54ba91e..07b23418 100644 --- a/examples/_config.yml +++ b/examples/_config.yml @@ -12,7 +12,7 @@ sphinx: execute: # Exclude some examples from execution (these are still deployed as html pages) - exclude_patterns: [".venv/*", "Forest_portability_example.ipynb", "backends_example.ipynb", "qiskit_integration.ipynb", "comparing_simulators.ipynb", "expectation_value_example.ipynb", "pytket-qujax_heisenberg_vqe.ipynb", "spam_example.ipynb", "tket_benchmarking.ipynb", "entanglement_swapping.ipynb", "pytket-qujax-classification.ipynb"] + exclude_patterns: [".venv/*", "Forest_portability_example.ipynb", "backends_example.ipynb", "qiskit_integration.ipynb", "comparing_simulators.ipynb", "expectation_value_example.ipynb", "pytket-qujax_heisenberg_vqe.ipynb", "spam_example.ipynb", "entanglement_swapping.ipynb", "pytket-qujax-classification.ipynb"] timeout: 90 # The maximum time (in seconds) each notebook cell is allowed to run. # Information about where the book exists on the web From 10f1aa4281978c0a2bf4f5ca921688aad2909b1a Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Wed, 8 Nov 2023 13:54:08 +0000 Subject: [PATCH 45/51] Remove Other section from TOC, add Algorithms and Protocols --- examples/_toc.yml | 7 +++---- 1 file changed, 3 insertions(+), 4 deletions(-) diff --git a/examples/_toc.yml b/examples/_toc.yml index c2227cb8..226e66b6 100644 --- a/examples/_toc.yml +++ b/examples/_toc.yml @@ -24,16 +24,15 @@ parts: - file: ansatz_sequence_example - file: measurement_reduction_example - file: contextual_optimization - - caption: Algorithm Demos + - caption: Algorithms and Protocols chapters: - file: phase_estimation - file: ucc_vqe - file: pytket-qujax-classification - file: pytket-qujax_qaoa - file: pytket-qujax_heisenberg_vqe - - caption: Other - chapters: + - file: expectation_value_example - file: entanglement_swapping - file: spam_example - - file: expectation_value_example + chapters: - file: README From fa5d53420b523dfa456044729506a60be933523f Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Wed, 8 Nov 2023 14:06:49 +0000 Subject: [PATCH 46/51] add Contributing.md --- examples/{README.md => CONTRIBUTING.md} | 0 examples/_toc.yml | 3 ++- 2 files changed, 2 insertions(+), 1 deletion(-) rename examples/{README.md => CONTRIBUTING.md} (100%) diff --git a/examples/README.md b/examples/CONTRIBUTING.md similarity index 100% rename from examples/README.md rename to examples/CONTRIBUTING.md diff --git a/examples/_toc.yml b/examples/_toc.yml index 226e66b6..0c14abee 100644 --- a/examples/_toc.yml +++ b/examples/_toc.yml @@ -34,5 +34,6 @@ parts: - file: expectation_value_example - file: entanglement_swapping - file: spam_example + - caption: Contributing chapters: - - file: README + - file: CONTRIBUTING From 85f64b2dade611187a17b9dec737d005b4ea0017 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Wed, 8 Nov 2023 14:11:38 +0000 Subject: [PATCH 47/51] updates notebook titles --- examples/entanglement_swapping.ipynb | 2 +- examples/python/entanglement_swapping.py | 2 +- examples/python/ucc_vqe.py | 2 +- examples/ucc_vqe.ipynb | 2 +- 4 files changed, 4 insertions(+), 4 deletions(-) diff --git a/examples/entanglement_swapping.ipynb b/examples/entanglement_swapping.ipynb index 35896a50..b5dfd1ad 100644 --- a/examples/entanglement_swapping.ipynb +++ b/examples/entanglement_swapping.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# Iterated entanglement swapping using TKET"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In this tutorial, we will focus on:
\n", "- designing circuits with mid-circuit measurement and conditional gates;
\n", "- utilising noise models in supported simulators."]}, {"cell_type": "markdown", "metadata": {}, "source": ["This example assumes the reader is familiar with the Qubit Teleportation and Entanglement Swapping protocols, and basic models of noise in quantum devices.
\n", "
\n", "To run this example, you will need `pytket`, `pytket-qiskit`, and `plotly` (installed via `pip`). To view the graphs, you will need an intallation of `plotly-orca`.
\n", "
\n", "Current quantum hardware fits into the NISQ (Noisy, Intermediate-Scale Quantum) regime. This noise cannot realistically be combatted using conventional error correcting codes, because of the lack of available qubits, noise levels exceeding the code thresholds, and very few devices available that can perform measurements and corrections mid-circuit. Analysis of how quantum algorithms perform under noisy conditions is a very active research area, as is finding ways to cope with it. Here, we will look at how well we can perform the Entanglement Swapping protocol with different noise levels.
\n", "
\n", "The Entanglement Swapping protocol requires two parties to share Bell pairs with a third party, who applies the Qubit Teleportation protocol to generate a Bell pair between the two parties. The Qubit Teleportation step requires us to be able to measure some qubits and make subsequent corrections to the remaining qubits. There are only a handful of simulators and devices that currently support this, with others restricted to only measuring the qubits at the end of the circuit.
\n", "
\n", "The most popular circuit model with conditional gates at the moment is that provided by the OpenQASM language. This permits a very restricted model of classical logic, where we can apply a gate conditionally on the exact value of a classical register. There is no facility in the current spec for Boolean logic or classical operations to apply any function to the value prior to the equality check. For example, Qubit Teleportation can be performed by the following QASM:
\n", "`OPENQASM 2.0;`
\n", "`include \"qelib1.inc\";`
\n", "`qreg a[2];`
\n", "`qreg b[1];`
\n", "`creg c[2];`
\n", "`// Bell state between Alice and Bob`
\n", "`h a[1];`
\n", "`cx a[1],b[0];`
\n", "`// Bell measurement of Alice's qubits`
\n", "`cx a[0],a[1];`
\n", "`h a[0];`
\n", "`measure a[0] -> c[0];`
\n", "`measure a[1] -> c[1];`
\n", "`// Correction of Bob's qubit`
\n", "`if(c==1) z b[0];`
\n", "`if(c==3) z b[0];`
\n", "`if(c==2) x b[0];`
\n", "`if(c==3) x b[0];`
\n", "
\n", "This corresponds to the following `pytket` code:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qtel = Circuit()\n", "alice = qtel.add_q_register(\"a\", 2)\n", "bob = qtel.add_q_register(\"b\", 1)\n", "data = qtel.add_c_register(\"d\", 2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Bell state between Alice and Bob:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qtel.H(alice[1])\n", "qtel.CX(alice[1], bob[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Bell measurement of Alice's qubits:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qtel.CX(alice[0], alice[1])\n", "qtel.H(alice[0])\n", "qtel.Measure(alice[0], data[0])\n", "qtel.Measure(alice[1], data[1])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Correction of Bob's qubit:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["qtel.X(bob[0], condition_bits=[data[0], data[1]], condition_value=2)\n", "qtel.X(bob[0], condition_bits=[data[0], data[1]], condition_value=3)\n", "qtel.Z(bob[0], condition_bits=[data[0], data[1]], condition_value=1)\n", "qtel.Z(bob[0], condition_bits=[data[0], data[1]], condition_value=3)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["So to demonstrate the Entanglement Swapping protocol, we just need to run this on one side of a Bell pair."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["es = Circuit()\n", "ava = es.add_q_register(\"a\", 1)\n", "bella = es.add_q_register(\"b\", 2)\n", "charlie = es.add_q_register(\"c\", 1)\n", "data = es.add_c_register(\"d\", 2)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Bell state between Ava and Bella:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["es.H(ava[0])\n", "es.CX(ava[0], bella[0])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Teleport `bella[0]` to `charlie[0]`:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["tel_to_c = qtel.copy()\n", "tel_to_c.rename_units({alice[0]: bella[0], alice[1]: bella[1], bob[0]: charlie[0]})\n", "es.append(tel_to_c)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(es.get_commands())"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Let's start by running a noiseless simulation of this to verify that what we get looks like a Bell pair."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import AerBackend"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Connect to a simulator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Make a ZZ measurement of the Bell pair:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["bell_test = es.copy()\n", "bell_test.Measure(ava[0], data[0])\n", "bell_test.Measure(charlie[0], data[1])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Run the experiment:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["bell_test = backend.get_compiled_circuit(bell_test)\n", "from pytket.circuit.display import render_circuit_jupyter"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["render_circuit_jupyter(bell_test)\n", "handle = backend.process_circuit(bell_test, n_shots=2000)\n", "counts = backend.get_result(handle).get_counts()"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(counts)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is good, we have got roughly 50/50 measurement results of 00 and 11 under the ZZ operator. But there are many other states beyond the Bell state that also generate this distribution, so to gain more confidence in our claim about the state we should make more measurements that also characterise it, i.e. perform state tomography.
\n", "
\n", "Here, we will demonstrate a naive approach to tomography that makes 3^n measurement circuits for an n-qubit state. More elaborate methods also exist."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.utils import append_pauli_measurement, probs_from_counts\n", "from itertools import product\n", "from scipy.linalg import lstsq, eigh\n", "import numpy as np"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def gen_tomography_circuits(state, qubits, bits):\n", " # Yields {X, Y, Z}^n measurements in lexicographical order\n", " # Only measures qubits, storing the result in bits\n", " # (since we don't care about the ancilla qubits)\n", " assert len(qubits) == len(bits)\n", " for paulis in product([Pauli.X, Pauli.Y, Pauli.Z], repeat=len(qubits)):\n", " circ = state.copy()\n", " for qb, b, p in zip(qubits, bits, paulis):\n", " if p == Pauli.X:\n", " circ.H(qb)\n", " elif p == Pauli.Y:\n", " circ.V(qb)\n", " circ.Measure(qb, b)\n", " yield circ"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def run_tomography_circuits(state, qubits, bits, backend):\n", " circs = list(gen_tomography_circuits(state, qubits, bits))\n", " # Compile and run each circuit\n", " circs = backend.get_compiled_circuits(circs)\n", " handles = backend.process_circuits(circs, n_shots=2000)\n", " # Get the observed measurement probabilities\n", " probs_list = []\n", " for result in backend.get_results(handles):\n", " counts = result.get_counts()\n", " probs = probs_from_counts(counts)\n", " probs_list.append(probs)\n", " return probs_list"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def fit_tomography_outcomes(probs_list, n_qbs):\n", " # Define the density matrices for the basis states\n", " basis = dict()\n", " basis[(Pauli.X, 0)] = np.asarray([[0.5, 0.5], [0.5, 0.5]])\n", " basis[(Pauli.X, 1)] = np.asarray([[0.5, -0.5], [-0.5, 0.5]])\n", " basis[(Pauli.Y, 0)] = np.asarray([[0.5, -0.5j], [0.5j, 0.5]])\n", " basis[(Pauli.Y, 1)] = np.asarray([[0.5, 0.5j], [-0.5j, 0.5]])\n", " basis[(Pauli.Z, 0)] = np.asarray([[1, 0], [0, 0]])\n", " basis[(Pauli.Z, 1)] = np.asarray([[0, 0], [0, 1]])\n", " dim = 2**n_qbs\n", " # Define vector all_probs as a concatenation of probability vectors for each measurement (2**n x 3**n, 1)\n", " # Define matrix all_ops mapping a (vectorised) density matrix to a vector of probabilities for each measurement\n", " # (2**n x 3**n, 2**n x 2**n)\n", " all_probs = []\n", " all_ops = []\n", " for paulis, probs in zip(\n", " product([Pauli.X, Pauli.Y, Pauli.Z], repeat=n_qbs), probs_list\n", " ):\n", " prob_vec = []\n", " meas_ops = []\n", " for outcome in product([0, 1], repeat=n_qbs):\n", " prob_vec.append(probs.get(outcome, 0))\n", " op = np.eye(1, dtype=complex)\n", " for p, o in zip(paulis, outcome):\n", " op = np.kron(op, basis[(p, o)])\n", " meas_ops.append(op.reshape(1, dim * dim).conj())\n", " all_probs.append(np.vstack(prob_vec))\n", " all_ops.append(np.vstack(meas_ops))\n", " # Solve for density matrix by minimising || all_ops * dm - all_probs ||\n", " dm, _, _, _ = lstsq(np.vstack(all_ops), np.vstack(all_probs))\n", " dm = dm.reshape(dim, dim)\n", " # Make density matrix positive semi-definite\n", " v, w = eigh(dm)\n", " for i in range(dim):\n", " if v[i] < 0:\n", " for j in range(i + 1, dim):\n", " v[j] += v[i] / (dim - (i + 1))\n", " v[i] = 0\n", " dm = np.zeros([dim, dim], dtype=complex)\n", " for j in range(dim):\n", " dm += v[j] * np.outer(w[:, j], np.conj(w[:, j]))\n", " # Normalise trace of density matrix\n", " dm /= np.trace(dm)\n", " return dm"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["probs_list = run_tomography_circuits(\n", " es, [ava[0], charlie[0]], [data[0], data[1]], backend\n", ")\n", "dm = fit_tomography_outcomes(probs_list, 2)\n", "print(dm.round(3))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is very close to the true density matrix for a pure Bell state. We can attribute the error here to the sampling error since we only take 2000 samples of each measurement circuit.
\n", "
\n", "To quantify exactly how similar it is to the correct density matrix, we can calculate the fidelity."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from scipy.linalg import sqrtm"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def fidelity(dm0, dm1):\n", " # Calculate the fidelity between two density matrices\n", " sq0 = sqrtm(dm0)\n", " sq1 = sqrtm(dm1)\n", " return np.linalg.norm(sq0.dot(sq1)) ** 2"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["bell_state = np.asarray(\n", " [\n", " [0.5, 0, 0, 0.5],\n", " [0, 0, 0, 0],\n", " [0, 0, 0, 0],\n", " [0.5, 0, 0, 0.5],\n", " ]\n", ")\n", "print(fidelity(dm, bell_state))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This high fidelity is unsurprising since we have a completely noiseless simulation. So the next step is to add some noise to the simulation and observe how the overall fidelity is affected. The `AerBackend` wraps around the Qiskit Aer simulator and can pass on any `qiskit.providers.aer.noise.NoiseModel` to the simulator. Let's start by adding some uniform depolarising noise to each CX gate and some uniform measurement error."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from qiskit.providers.aer.noise import NoiseModel, depolarizing_error, ReadoutError"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def make_noise_model(dep_err_rate, ro_err_rate, qubits):\n", " # Define a noise model that applies uniformly to the given qubits\n", " model = NoiseModel()\n", " dep_err = depolarizing_error(dep_err_rate, 2)\n", " ro_err = ReadoutError(\n", " [[1 - ro_err_rate, ro_err_rate], [ro_err_rate, 1 - ro_err_rate]]\n", " )\n", " # Add depolarising error to CX gates between any qubits (implying full connectivity)\n", " for i, j in product(qubits, repeat=2):\n", " if i != j:\n", " model.add_quantum_error(dep_err, [\"cx\"], [i, j])\n", " # Add readout error for each qubit\n", " for i in qubits:\n", " model.add_readout_error(ro_err, qubits=[i])\n", " return model"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_model = make_noise_model(0.03, 0.05, range(4))\n", "backend = AerBackend(noise_model=test_model)\n", "probs_list = run_tomography_circuits(\n", " es, [ava[0], charlie[0]], [data[0], data[1]], backend\n", ")\n", "dm = fit_tomography_outcomes(probs_list, 2)\n", "print(dm.round(3))\n", "print(fidelity(dm, bell_state))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Despite the very small circuit and the relatively small error rates, the fidelity of the final state has reduced considerably.
\n", "
\n", "As far as circuits go, the entanglement swapping protocol is little more than a toy example and is nothing close to the scale of circuits for most interesting quantum computational problems. However, it is possible to iterate the protocol many times to build up a larger computation, allowing us to see the impact of the noise at different scales."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import OpType\n", "from plotly.graph_objects import Scatter, Figure"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def iterated_entanglement_swap(n_iter):\n", " # Iterate the entanglement swapping protocol n_iter times\n", " it_es = Circuit()\n", " ava = it_es.add_q_register(\"a\", 1)\n", " bella = it_es.add_q_register(\"b\", 2)\n", " charlie = it_es.add_q_register(\"c\", 1)\n", " data = it_es.add_c_register(\"d\", 2)\n\n", " # Start with an initial Bell state\n", " it_es.H(ava[0])\n", " it_es.CX(ava[0], bella[0])\n", " for i in range(n_iter):\n", " if i % 2 == 0:\n", " # Teleport bella[0] to charlie[0] to give a Bell pair between ava[0] and charlier[0]\n", " tel_to_c = qtel.copy()\n", " tel_to_c.rename_units(\n", " {alice[0]: bella[0], alice[1]: bella[1], bob[0]: charlie[0]}\n", " )\n", " it_es.append(tel_to_c)\n", " it_es.add_gate(OpType.Reset, [bella[0]])\n", " it_es.add_gate(OpType.Reset, [bella[1]])\n", " else:\n", " # Teleport charlie[0] to bella[0] to give a Bell pair between ava[0] and bella[0]\n", " tel_to_b = qtel.copy()\n", " tel_to_b.rename_units(\n", " {alice[0]: charlie[0], alice[1]: bella[1], bob[0]: bella[0]}\n", " )\n", " it_es.append(tel_to_b)\n", " it_es.add_gate(OpType.Reset, [bella[1]])\n", " it_es.add_gate(OpType.Reset, [charlie[0]])\n", " # Return the circuit and the qubits expected to share a Bell pair\n", " if n_iter % 2 == 0:\n", " return it_es, [ava[0], bella[0]]\n", " else:\n", " return it_es, [ava[0], charlie[0]]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def iterated_noisy_experiment(dep_err_rate, ro_err_rate, max_iter):\n", " # Set up the noisy simulator with the given error rates\n", " test_model = make_noise_model(dep_err_rate, ro_err_rate, range(4))\n", " backend = AerBackend(noise_model=test_model)\n", " # Estimate the fidelity after n iterations, from 0 to max_iter (inclusive)\n", " fid_list = []\n", " for i in range(max_iter + 1):\n", " it_es, qubits = iterated_entanglement_swap(i)\n", " probs_list = run_tomography_circuits(it_es, qubits, [data[0], data[1]], backend)\n", " dm = fit_tomography_outcomes(probs_list, 2)\n", " fid = fidelity(dm, bell_state)\n", " fid_list.append(fid)\n", " return fid_list"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["fig = Figure()\n", "fig.update_layout(\n", " title=\"Iterated Entanglement Swapping under Noise (dep_err = 0.03)\",\n", " xaxis_title=\"Iterations\",\n", " xaxis=dict(range=[0, 10]),\n", " yaxis_title=\"Fidelity\",\n", ")\n", "iter_range = np.arange(11)\n", "for i in range(7):\n", " fids = iterated_noisy_experiment(0.03, 0.025 * i, 10)\n", " plot_data = Scatter(\n", " x=iter_range, y=fids, name=\"ro_err=\" + str(np.round(0.025 * i, 3))\n", " )\n", " fig.add_trace(plot_data)\n", "try:\n", " fig.show(renderer=\"svg\")\n", "except ValueError as e:\n", " print(e) # requires plotly-orca"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["fig = Figure()\n", "fig.update_layout(\n", " title=\"Iterated Entanglement Swapping under Noise (ro_err = 0.05)\",\n", " xaxis_title=\"Iterations\",\n", " xaxis=dict(range=[0, 10]),\n", " yaxis_title=\"Fidelity\",\n", ")\n", "iter_range = np.arange(11)\n", "for i in range(9):\n", " fids = iterated_noisy_experiment(0.01 * i, 0.05, 10)\n", " plot_data = Scatter(\n", " x=iter_range, y=fids, name=\"dep_err=\" + str(np.round(0.01 * i, 3))\n", " )\n", " fig.add_trace(plot_data)\n", "try:\n", " fig.show(renderer=\"svg\")\n", "except ValueError as e:\n", " print(e) # requires plotly-orca"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These graphs are not very surprising, but are still important for seeing that the current error rates of typical NISQ devices become crippling for fidelities very quickly after repeated mid-circuit measurements and corrections (even with this overly-simplified model with uniform noise and no crosstalk or higher error modes). This provides good motivation for the adoption of error mitigation techniques, and for the development of new techniques that are robust to errors in mid-circuit measurements."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Exercises:
\n", "- Vary the fixed noise levels to compare how impactful the depolarising and measurement errors are.
\n", "- Add extra noise characteristics to the noise model to obtain something that more resembles a real device. Possible options include adding error during the reset operations, extending the errors to be non-local, or constructing the noise model from a device's calibration data.
\n", "- Change the circuit from iterated entanglement swapping to iterated applications of a correction circuit from a simple error-correcting code. Do you expect this to be more sensitive to depolarising errors from unitary gates or measurement errors?"]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells":[{"cell_type":"markdown","metadata":{},"source":["# Iterated entanglement swapping"]},{"cell_type":"markdown","metadata":{},"source":["In this tutorial, we will focus on:\n","- designing circuits with mid-circuit measurement and conditional gates;\n","- utilising noise models in supported simulators."]},{"cell_type":"markdown","metadata":{},"source":["This example assumes the reader is familiar with the Qubit Teleportation and Entanglement Swapping protocols, and basic models of noise in quantum devices.\n","\n","To run this example, you will need `pytket`, `pytket-qiskit`, and `plotly` (installed via `pip`). To view the graphs, you will need an intallation of `plotly-orca`.\n","\n","Current quantum hardware fits into the NISQ (Noisy, Intermediate-Scale Quantum) regime. This noise cannot realistically be combatted using conventional error correcting codes, because of the lack of available qubits, noise levels exceeding the code thresholds, and very few devices available that can perform measurements and corrections mid-circuit. Analysis of how quantum algorithms perform under noisy conditions is a very active research area, as is finding ways to cope with it. Here, we will look at how well we can perform the Entanglement Swapping protocol with different noise levels.\n","\n","The Entanglement Swapping protocol requires two parties to share Bell pairs with a third party, who applies the Qubit Teleportation protocol to generate a Bell pair between the two parties. The Qubit Teleportation step requires us to be able to measure some qubits and make subsequent corrections to the remaining qubits. There are only a handful of simulators and devices that currently support this, with others restricted to only measuring the qubits at the end of the circuit.\n","\n","The most popular circuit model with conditional gates at the moment is that provided by the OpenQASM language. This permits a very restricted model of classical logic, where we can apply a gate conditionally on the exact value of a classical register. There is no facility in the current spec for Boolean logic or classical operations to apply any function to the value prior to the equality check. For example, Qubit Teleportation can be performed by the following QASM:\n","`OPENQASM 2.0;`\n","`include \"qelib1.inc\";`\n","`qreg a[2];`\n","`qreg b[1];`\n","`creg c[2];`\n","`// Bell state between Alice and Bob`\n","`h a[1];`\n","`cx a[1],b[0];`\n","`// Bell measurement of Alice's qubits`\n","`cx a[0],a[1];`\n","`h a[0];`\n","`measure a[0] -> c[0];`\n","`measure a[1] -> c[1];`\n","`// Correction of Bob's qubit`\n","`if(c==1) z b[0];`\n","`if(c==3) z b[0];`\n","`if(c==2) x b[0];`\n","`if(c==3) x b[0];`\n","\n","This corresponds to the following `pytket` code:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket import Circuit"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qtel = Circuit()\n","alice = qtel.add_q_register(\"a\", 2)\n","bob = qtel.add_q_register(\"b\", 1)\n","data = qtel.add_c_register(\"d\", 2)"]},{"cell_type":"markdown","metadata":{},"source":["Bell state between Alice and Bob:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qtel.H(alice[1])\n","qtel.CX(alice[1], bob[0])"]},{"cell_type":"markdown","metadata":{},"source":["Bell measurement of Alice's qubits:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qtel.CX(alice[0], alice[1])\n","qtel.H(alice[0])\n","qtel.Measure(alice[0], data[0])\n","qtel.Measure(alice[1], data[1])"]},{"cell_type":"markdown","metadata":{},"source":["Correction of Bob's qubit:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["qtel.X(bob[0], condition_bits=[data[0], data[1]], condition_value=2)\n","qtel.X(bob[0], condition_bits=[data[0], data[1]], condition_value=3)\n","qtel.Z(bob[0], condition_bits=[data[0], data[1]], condition_value=1)\n","qtel.Z(bob[0], condition_bits=[data[0], data[1]], condition_value=3)"]},{"cell_type":"markdown","metadata":{},"source":["So to demonstrate the Entanglement Swapping protocol, we just need to run this on one side of a Bell pair."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["es = Circuit()\n","ava = es.add_q_register(\"a\", 1)\n","bella = es.add_q_register(\"b\", 2)\n","charlie = es.add_q_register(\"c\", 1)\n","data = es.add_c_register(\"d\", 2)"]},{"cell_type":"markdown","metadata":{},"source":["Bell state between Ava and Bella:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["es.H(ava[0])\n","es.CX(ava[0], bella[0])"]},{"cell_type":"markdown","metadata":{},"source":["Teleport `bella[0]` to `charlie[0]`:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["tel_to_c = qtel.copy()\n","tel_to_c.rename_units({alice[0]: bella[0], alice[1]: bella[1], bob[0]: charlie[0]})\n","es.append(tel_to_c)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(es.get_commands())"]},{"cell_type":"markdown","metadata":{},"source":["Let's start by running a noiseless simulation of this to verify that what we get looks like a Bell pair."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.extensions.qiskit import AerBackend"]},{"cell_type":"markdown","metadata":{},"source":["Connect to a simulator:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["backend = AerBackend()"]},{"cell_type":"markdown","metadata":{},"source":["Make a ZZ measurement of the Bell pair:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["bell_test = es.copy()\n","bell_test.Measure(ava[0], data[0])\n","bell_test.Measure(charlie[0], data[1])"]},{"cell_type":"markdown","metadata":{},"source":["Run the experiment:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["bell_test = backend.get_compiled_circuit(bell_test)\n","from pytket.circuit.display import render_circuit_jupyter"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["render_circuit_jupyter(bell_test)\n","handle = backend.process_circuit(bell_test, n_shots=2000)\n","counts = backend.get_result(handle).get_counts()"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(counts)"]},{"cell_type":"markdown","metadata":{},"source":["This is good, we have got roughly 50/50 measurement results of 00 and 11 under the ZZ operator. But there are many other states beyond the Bell state that also generate this distribution, so to gain more confidence in our claim about the state we should make more measurements that also characterise it, i.e. perform state tomography.\n","\n","Here, we will demonstrate a naive approach to tomography that makes 3^n measurement circuits for an n-qubit state. More elaborate methods also exist."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket.pauli import Pauli, QubitPauliString\n","from pytket.utils import append_pauli_measurement, probs_from_counts\n","from itertools import product\n","from scipy.linalg import lstsq, eigh\n","import numpy as np"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def gen_tomography_circuits(state, qubits, bits):\n"," # Yields {X, Y, Z}^n measurements in lexicographical order\n"," # Only measures qubits, storing the result in bits\n"," # (since we don't care about the ancilla qubits)\n"," assert len(qubits) == len(bits)\n"," for paulis in product([Pauli.X, Pauli.Y, Pauli.Z], repeat=len(qubits)):\n"," circ = state.copy()\n"," for qb, b, p in zip(qubits, bits, paulis):\n"," if p == Pauli.X:\n"," circ.H(qb)\n"," elif p == Pauli.Y:\n"," circ.V(qb)\n"," circ.Measure(qb, b)\n"," yield circ"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def run_tomography_circuits(state, qubits, bits, backend):\n"," circs = list(gen_tomography_circuits(state, qubits, bits))\n"," # Compile and run each circuit\n"," circs = backend.get_compiled_circuits(circs)\n"," handles = backend.process_circuits(circs, n_shots=2000)\n"," # Get the observed measurement probabilities\n"," probs_list = []\n"," for result in backend.get_results(handles):\n"," counts = result.get_counts()\n"," probs = probs_from_counts(counts)\n"," probs_list.append(probs)\n"," return probs_list"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def fit_tomography_outcomes(probs_list, n_qbs):\n"," # Define the density matrices for the basis states\n"," basis = dict()\n"," basis[(Pauli.X, 0)] = np.asarray([[0.5, 0.5], [0.5, 0.5]])\n"," basis[(Pauli.X, 1)] = np.asarray([[0.5, -0.5], [-0.5, 0.5]])\n"," basis[(Pauli.Y, 0)] = np.asarray([[0.5, -0.5j], [0.5j, 0.5]])\n"," basis[(Pauli.Y, 1)] = np.asarray([[0.5, 0.5j], [-0.5j, 0.5]])\n"," basis[(Pauli.Z, 0)] = np.asarray([[1, 0], [0, 0]])\n"," basis[(Pauli.Z, 1)] = np.asarray([[0, 0], [0, 1]])\n"," dim = 2**n_qbs\n"," # Define vector all_probs as a concatenation of probability vectors for each measurement (2**n x 3**n, 1)\n"," # Define matrix all_ops mapping a (vectorised) density matrix to a vector of probabilities for each measurement\n"," # (2**n x 3**n, 2**n x 2**n)\n"," all_probs = []\n"," all_ops = []\n"," for paulis, probs in zip(\n"," product([Pauli.X, Pauli.Y, Pauli.Z], repeat=n_qbs), probs_list\n"," ):\n"," prob_vec = []\n"," meas_ops = []\n"," for outcome in product([0, 1], repeat=n_qbs):\n"," prob_vec.append(probs.get(outcome, 0))\n"," op = np.eye(1, dtype=complex)\n"," for p, o in zip(paulis, outcome):\n"," op = np.kron(op, basis[(p, o)])\n"," meas_ops.append(op.reshape(1, dim * dim).conj())\n"," all_probs.append(np.vstack(prob_vec))\n"," all_ops.append(np.vstack(meas_ops))\n"," # Solve for density matrix by minimising || all_ops * dm - all_probs ||\n"," dm, _, _, _ = lstsq(np.vstack(all_ops), np.vstack(all_probs))\n"," dm = dm.reshape(dim, dim)\n"," # Make density matrix positive semi-definite\n"," v, w = eigh(dm)\n"," for i in range(dim):\n"," if v[i] < 0:\n"," for j in range(i + 1, dim):\n"," v[j] += v[i] / (dim - (i + 1))\n"," v[i] = 0\n"," dm = np.zeros([dim, dim], dtype=complex)\n"," for j in range(dim):\n"," dm += v[j] * np.outer(w[:, j], np.conj(w[:, j]))\n"," # Normalise trace of density matrix\n"," dm /= np.trace(dm)\n"," return dm"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["probs_list = run_tomography_circuits(\n"," es, [ava[0], charlie[0]], [data[0], data[1]], backend\n",")\n","dm = fit_tomography_outcomes(probs_list, 2)\n","print(dm.round(3))"]},{"cell_type":"markdown","metadata":{},"source":["This is very close to the true density matrix for a pure Bell state. We can attribute the error here to the sampling error since we only take 2000 samples of each measurement circuit.\n","\n","To quantify exactly how similar it is to the correct density matrix, we can calculate the fidelity."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from scipy.linalg import sqrtm"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def fidelity(dm0, dm1):\n"," # Calculate the fidelity between two density matrices\n"," sq0 = sqrtm(dm0)\n"," sq1 = sqrtm(dm1)\n"," return np.linalg.norm(sq0.dot(sq1)) ** 2"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["bell_state = np.asarray(\n"," [\n"," [0.5, 0, 0, 0.5],\n"," [0, 0, 0, 0],\n"," [0, 0, 0, 0],\n"," [0.5, 0, 0, 0.5],\n"," ]\n",")\n","print(fidelity(dm, bell_state))"]},{"cell_type":"markdown","metadata":{},"source":["This high fidelity is unsurprising since we have a completely noiseless simulation. So the next step is to add some noise to the simulation and observe how the overall fidelity is affected. The `AerBackend` wraps around the Qiskit Aer simulator and can pass on any `qiskit.providers.aer.noise.NoiseModel` to the simulator. Let's start by adding some uniform depolarising noise to each CX gate and some uniform measurement error."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from qiskit.providers.aer.noise import NoiseModel, depolarizing_error, ReadoutError"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def make_noise_model(dep_err_rate, ro_err_rate, qubits):\n"," # Define a noise model that applies uniformly to the given qubits\n"," model = NoiseModel()\n"," dep_err = depolarizing_error(dep_err_rate, 2)\n"," ro_err = ReadoutError(\n"," [[1 - ro_err_rate, ro_err_rate], [ro_err_rate, 1 - ro_err_rate]]\n"," )\n"," # Add depolarising error to CX gates between any qubits (implying full connectivity)\n"," for i, j in product(qubits, repeat=2):\n"," if i != j:\n"," model.add_quantum_error(dep_err, [\"cx\"], [i, j])\n"," # Add readout error for each qubit\n"," for i in qubits:\n"," model.add_readout_error(ro_err, qubits=[i])\n"," return model"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["test_model = make_noise_model(0.03, 0.05, range(4))\n","backend = AerBackend(noise_model=test_model)\n","probs_list = run_tomography_circuits(\n"," es, [ava[0], charlie[0]], [data[0], data[1]], backend\n",")\n","dm = fit_tomography_outcomes(probs_list, 2)\n","print(dm.round(3))\n","print(fidelity(dm, bell_state))"]},{"cell_type":"markdown","metadata":{},"source":["Despite the very small circuit and the relatively small error rates, the fidelity of the final state has reduced considerably.\n","\n","As far as circuits go, the entanglement swapping protocol is little more than a toy example and is nothing close to the scale of circuits for most interesting quantum computational problems. However, it is possible to iterate the protocol many times to build up a larger computation, allowing us to see the impact of the noise at different scales."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket import OpType\n","from plotly.graph_objects import Scatter, Figure"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def iterated_entanglement_swap(n_iter):\n"," # Iterate the entanglement swapping protocol n_iter times\n"," it_es = Circuit()\n"," ava = it_es.add_q_register(\"a\", 1)\n"," bella = it_es.add_q_register(\"b\", 2)\n"," charlie = it_es.add_q_register(\"c\", 1)\n"," data = it_es.add_c_register(\"d\", 2)\n","\n"," # Start with an initial Bell state\n"," it_es.H(ava[0])\n"," it_es.CX(ava[0], bella[0])\n"," for i in range(n_iter):\n"," if i % 2 == 0:\n"," # Teleport bella[0] to charlie[0] to give a Bell pair between ava[0] and charlier[0]\n"," tel_to_c = qtel.copy()\n"," tel_to_c.rename_units(\n"," {alice[0]: bella[0], alice[1]: bella[1], bob[0]: charlie[0]}\n"," )\n"," it_es.append(tel_to_c)\n"," it_es.add_gate(OpType.Reset, [bella[0]])\n"," it_es.add_gate(OpType.Reset, [bella[1]])\n"," else:\n"," # Teleport charlie[0] to bella[0] to give a Bell pair between ava[0] and bella[0]\n"," tel_to_b = qtel.copy()\n"," tel_to_b.rename_units(\n"," {alice[0]: charlie[0], alice[1]: bella[1], bob[0]: bella[0]}\n"," )\n"," it_es.append(tel_to_b)\n"," it_es.add_gate(OpType.Reset, [bella[1]])\n"," it_es.add_gate(OpType.Reset, [charlie[0]])\n"," # Return the circuit and the qubits expected to share a Bell pair\n"," if n_iter % 2 == 0:\n"," return it_es, [ava[0], bella[0]]\n"," else:\n"," return it_es, [ava[0], charlie[0]]"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def iterated_noisy_experiment(dep_err_rate, ro_err_rate, max_iter):\n"," # Set up the noisy simulator with the given error rates\n"," test_model = make_noise_model(dep_err_rate, ro_err_rate, range(4))\n"," backend = AerBackend(noise_model=test_model)\n"," # Estimate the fidelity after n iterations, from 0 to max_iter (inclusive)\n"," fid_list = []\n"," for i in range(max_iter + 1):\n"," it_es, qubits = iterated_entanglement_swap(i)\n"," probs_list = run_tomography_circuits(it_es, qubits, [data[0], data[1]], backend)\n"," dm = fit_tomography_outcomes(probs_list, 2)\n"," fid = fidelity(dm, bell_state)\n"," fid_list.append(fid)\n"," return fid_list"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["fig = Figure()\n","fig.update_layout(\n"," title=\"Iterated Entanglement Swapping under Noise (dep_err = 0.03)\",\n"," xaxis_title=\"Iterations\",\n"," xaxis=dict(range=[0, 10]),\n"," yaxis_title=\"Fidelity\",\n",")\n","iter_range = np.arange(11)\n","for i in range(7):\n"," fids = iterated_noisy_experiment(0.03, 0.025 * i, 10)\n"," plot_data = Scatter(\n"," x=iter_range, y=fids, name=\"ro_err=\" + str(np.round(0.025 * i, 3))\n"," )\n"," fig.add_trace(plot_data)\n","try:\n"," fig.show(renderer=\"svg\")\n","except ValueError as e:\n"," print(e) # requires plotly-orca"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["fig = Figure()\n","fig.update_layout(\n"," title=\"Iterated Entanglement Swapping under Noise (ro_err = 0.05)\",\n"," xaxis_title=\"Iterations\",\n"," xaxis=dict(range=[0, 10]),\n"," yaxis_title=\"Fidelity\",\n",")\n","iter_range = np.arange(11)\n","for i in range(9):\n"," fids = iterated_noisy_experiment(0.01 * i, 0.05, 10)\n"," plot_data = Scatter(\n"," x=iter_range, y=fids, name=\"dep_err=\" + str(np.round(0.01 * i, 3))\n"," )\n"," fig.add_trace(plot_data)\n","try:\n"," fig.show(renderer=\"svg\")\n","except ValueError as e:\n"," print(e) # requires plotly-orca"]},{"cell_type":"markdown","metadata":{},"source":["These graphs are not very surprising, but are still important for seeing that the current error rates of typical NISQ devices become crippling for fidelities very quickly after repeated mid-circuit measurements and corrections (even with this overly-simplified model with uniform noise and no crosstalk or higher error modes). This provides good motivation for the adoption of error mitigation techniques, and for the development of new techniques that are robust to errors in mid-circuit measurements."]},{"cell_type":"markdown","metadata":{},"source":["Exercises:\n","- Vary the fixed noise levels to compare how impactful the depolarising and measurement errors are.\n","- Add extra noise characteristics to the noise model to obtain something that more resembles a real device. Possible options include adding error during the reset operations, extending the errors to be non-local, or constructing the noise model from a device's calibration data.\n","- Change the circuit from iterated entanglement swapping to iterated applications of a correction circuit from a simple error-correcting code. Do you expect this to be more sensitive to depolarising errors from unitary gates or measurement errors?"]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} diff --git a/examples/python/entanglement_swapping.py b/examples/python/entanglement_swapping.py index 2d75d330..105d3bfc 100644 --- a/examples/python/entanglement_swapping.py +++ b/examples/python/entanglement_swapping.py @@ -1,4 +1,4 @@ -# # Iterated entanglement swapping using TKET +# # Iterated entanglement swapping # In this tutorial, we will focus on: # - designing circuits with mid-circuit measurement and conditional gates; diff --git a/examples/python/ucc_vqe.py b/examples/python/ucc_vqe.py index bb529be9..43b04287 100644 --- a/examples/python/ucc_vqe.py +++ b/examples/python/ucc_vqe.py @@ -1,4 +1,4 @@ -# # VQE with UCC ansatz using TKET +# # VQE with UCC ansatz # In this tutorial, we will focus on: # - building parameterised ansätze for variational algorithms; diff --git a/examples/ucc_vqe.ipynb b/examples/ucc_vqe.ipynb index b976650e..7ed8608d 100644 --- a/examples/ucc_vqe.ipynb +++ b/examples/ucc_vqe.ipynb @@ -1 +1 @@ -{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# VQE with UCC ansatz using TKET"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In this tutorial, we will focus on:
\n", "- building parameterised ans\u00e4tze for variational algorithms;
\n", "- compilation tools for UCC-style ans\u00e4tze."]}, {"cell_type": "markdown", "metadata": {}, "source": ["This example assumes the reader is familiar with the Variational Quantum Eigensolver and its application to electronic structure problems through the Unitary Coupled Cluster approach.
\n", "
\n", "To run this example, you will need `pytket` and `pytket-qiskit`, as well as `openfermion`, `scipy`, and `sympy`.
\n", "
\n", "We will start with a basic implementation and then gradually modify it to make it faster, more general, and less noisy. The final solution is given in full at the bottom of the notebook.
\n", "
\n", "Suppose we have some electronic configuration problem, expressed via a physical Hamiltonian. (The Hamiltonian and excitations in this example were obtained using `qiskit-aqua` version 0.5.2 and `pyscf` for H2, bond length 0.75A, sto3g basis, Jordan-Wigner encoding, with no qubit reduction or orbital freezing.). We express it succinctly using the openfermion library:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import openfermion as of"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian = (\n", " -0.8153001706270075 * of.QubitOperator(\"\")\n", " + 0.16988452027940318 * of.QubitOperator(\"Z0\")\n", " + -0.21886306781219608 * of.QubitOperator(\"Z1\")\n", " + 0.16988452027940323 * of.QubitOperator(\"Z2\")\n", " + -0.2188630678121961 * of.QubitOperator(\"Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z0 Z1\")\n", " + 0.16821198673715723 * of.QubitOperator(\"Z0 Z2\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z0 Z3\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z1 Z2\")\n", " + 0.1739537877649417 * of.QubitOperator(\"Z1 Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z2 Z3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 Y2 Y3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 Y2 Y3\")\n", ")\n", "nuclear_repulsion_energy = 0.70556961456"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We would like to define our ansatz for arbitrary parameter values. For simplicity, let's start with a Hardware Efficient Ansatz."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Hardware efficient ansatz:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def hea(params):\n", " ansatz = Circuit(4)\n", " for i in range(4):\n", " ansatz.Ry(params[i], i)\n", " for i in range(3):\n", " ansatz.CX(i, i + 1)\n", " for i in range(4):\n", " ansatz.Ry(params[4 + i], i)\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can use this to build the objective function for our optimisation."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import AerBackend\n", "from pytket.utils.expectations import expectation_from_counts"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Naive objective function:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " energy = 0\n", " for term, coeff in hamiltonian.terms.items():\n", " if not term:\n", " energy += coeff\n", " continue\n", " circ = hea(params)\n", " circ.add_c_register(\"c\", len(term))\n", " for i, (q, pauli) in enumerate(term):\n", " if pauli == \"X\":\n", " circ.H(q)\n", " elif pauli == \"Y\":\n", " circ.V(q)\n", " circ.Measure(q, i)\n", " compiled_circ = backend.get_compiled_circuit(circ)\n", " counts = backend.run_circuit(compiled_circ, n_shots=4000).get_counts()\n", " energy += coeff * expectation_from_counts(counts)\n", " return energy + nuclear_repulsion_energy"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This objective function is then run through a classical optimiser to find the set of parameter values that minimise the energy of the system. For the sake of example, we will just run this with a single parameter value."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["arg_values = [\n", " -7.31158201e-02,\n", " -1.64514836e-04,\n", " 1.12585591e-03,\n", " -2.58367544e-03,\n", " 1.00006068e00,\n", " -1.19551357e-03,\n", " 9.99963988e-01,\n", " 2.53283285e-03,\n", "]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["energy = objective(arg_values)\n", "print(energy)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The HEA is designed to cram as many orthogonal degrees of freedom into a small circuit as possible to be able to explore a large region of the Hilbert space whilst the circuits themselves can be run with minimal noise. These ans\u00e4tze give virtually-optimal circuits by design, but suffer from an excessive number of variational parameters making convergence slow, barren plateaus where the classical optimiser fails to make progress, and spanning a space where most states lack a physical interpretation. These drawbacks can necessitate adding penalties and may mean that the ansatz cannot actually express the true ground state.
\n", "
\n", "The UCC ansatz, on the other hand, is derived from the electronic configuration. It sacrifices efficiency of the circuit for the guarantee of physical states and the variational parameters all having some meaningful effect, which helps the classical optimisation to converge.
\n", "
\n", "This starts by defining the terms of our single and double excitations. These would usually be generated using the orbital configurations, so we will just use a hard-coded example here for the purposes of demonstration."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import Qubit\n", "from pytket.pauli import Pauli, QubitPauliString"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["q = [Qubit(i) for i in range(4)]\n", "xyii = QubitPauliString([q[0], q[1]], [Pauli.X, Pauli.Y])\n", "yxii = QubitPauliString([q[0], q[1]], [Pauli.Y, Pauli.X])\n", "iixy = QubitPauliString([q[2], q[3]], [Pauli.X, Pauli.Y])\n", "iiyx = QubitPauliString([q[2], q[3]], [Pauli.Y, Pauli.X])\n", "xxxy = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.X, Pauli.Y])\n", "xxyx = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.Y, Pauli.X])\n", "xyxx = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.X, Pauli.X])\n", "yxxx = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.X, Pauli.X])\n", "yyyx = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.Y, Pauli.X])\n", "yyxy = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.X, Pauli.Y])\n", "yxyy = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.Y, Pauli.Y])\n", "xyyy = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.Y, Pauli.Y])"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["singles_a = {xyii: 1.0, yxii: -1.0}\n", "singles_b = {iixy: 1.0, iiyx: -1.0}\n", "doubles = {\n", " xxxy: 0.25,\n", " xxyx: -0.25,\n", " xyxx: 0.25,\n", " yxxx: -0.25,\n", " yyyx: -0.25,\n", " yyxy: 0.25,\n", " yxyy: -0.25,\n", " xyyy: 0.25,\n", "}"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Building the ansatz circuit itself is often done naively by defining the map from each term down to basic gates and then applying it to each term."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def add_operator_term(circuit: Circuit, term: QubitPauliString, angle: float):\n", " qubits = []\n", " for q, p in term.map.items():\n", " if p != Pauli.I:\n", " qubits.append(q)\n", " if p == Pauli.X:\n", " circuit.H(q)\n", " elif p == Pauli.Y:\n", " circuit.V(q)\n", " for i in range(len(qubits) - 1):\n", " circuit.CX(i, i + 1)\n", " circuit.Rz(angle, len(qubits) - 1)\n", " for i in reversed(range(len(qubits) - 1)):\n", " circuit.CX(i, i + 1)\n", " for q, p in term.map.items():\n", " if p == Pauli.X:\n", " circuit.H(q)\n", " elif p == Pauli.Y:\n", " circuit.Vdg(q)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Unitary Coupled Cluster Singles & Doubles ansatz:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def ucc(params):\n", " ansatz = Circuit(4)\n", " # Set initial reference state\n", " ansatz.X(1).X(3)\n", " # Evolve by excitations\n", " for term, coeff in singles_a.items():\n", " add_operator_term(ansatz, term, coeff * params[0])\n", " for term, coeff in singles_b.items():\n", " add_operator_term(ansatz, term, coeff * params[1])\n", " for term, coeff in doubles.items():\n", " add_operator_term(ansatz, term, coeff * params[2])\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is already quite verbose, but `pytket` has a neat shorthand construction for these operator terms using the `PauliExpBox` construction. We can then decompose these into basic gates using the `DecomposeBoxes` compiler pass."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import PauliExpBox\n", "from pytket.passes import DecomposeBoxes"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def add_excitation(circ, term_dict, param):\n", " for term, coeff in term_dict.items():\n", " qubits, paulis = zip(*term.map.items())\n", " pbox = PauliExpBox(paulis, coeff * param)\n", " circ.add_pauliexpbox(pbox, qubits)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["UCC ansatz with syntactic shortcuts:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def ucc(params):\n", " ansatz = Circuit(4)\n", " ansatz.X(1).X(3)\n", " add_excitation(ansatz, singles_a, params[0])\n", " add_excitation(ansatz, singles_b, params[1])\n", " add_excitation(ansatz, doubles, params[2])\n", " DecomposeBoxes().apply(ansatz)\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The objective function can also be simplified using a utility method for constructing the measurement circuits and processing for expectation value calculations. For that, we convert the Hamiltonian to a pytket QubitPauliOperator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["pauli_sym = {\"I\": Pauli.I, \"X\": Pauli.X, \"Y\": Pauli.Y, \"Z\": Pauli.Z}"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qps_from_openfermion(paulis):\n", " \"\"\"Convert OpenFermion tensor of Paulis to pytket QubitPauliString.\"\"\"\n", " qlist = []\n", " plist = []\n", " for q, p in paulis:\n", " qlist.append(Qubit(q))\n", " plist.append(pauli_sym[p])\n", " return QubitPauliString(qlist, plist)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qpo_from_openfermion(openf_op):\n", " \"\"\"Convert OpenFermion QubitOperator to pytket QubitPauliOperator.\"\"\"\n", " tk_op = dict()\n", " for term, coeff in openf_op.terms.items():\n", " string = qps_from_openfermion(term)\n", " tk_op[string] = coeff\n", " return QubitPauliOperator(tk_op)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian_op = qpo_from_openfermion(hamiltonian)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Simplified objective function using utilities:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils.expectations import get_operator_expectation_value"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc(params)\n", " return (\n", " get_operator_expectation_value(circ, hamiltonian_op, backend, n_shots=4000)\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["arg_values = [-3.79002933e-05, 2.42964799e-05, 4.63447157e-01]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["energy = objective(arg_values)\n", "print(energy)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is now the simplest form that this operation can take, but it isn't necessarily the most effective. When we decompose the ansatz circuit into basic gates, it is still very expensive. We can employ some of the circuit simplification passes available in `pytket` to reduce its size and improve fidelity in practice.
\n", "
\n", "A good example is to decompose each `PauliExpBox` into basic gates and then apply `FullPeepholeOptimise`, which defines a compilation strategy utilising all of the simplifications in `pytket` that act locally on small regions of a circuit. We can examine the effectiveness by looking at the number of two-qubit gates before and after simplification, which tends to be a good indicator of fidelity for near-term systems where these gates are often slow and inaccurate."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import OpType\n", "from pytket.passes import FullPeepholeOptimise"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_circuit = ucc(arg_values)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count before\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth before\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["FullPeepholeOptimise().apply(test_circuit)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count after FPO\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth after FPO\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These simplification techniques are very general and are almost always beneficial to apply to a circuit if you want to eliminate local redundancies. But UCC ans\u00e4tze have extra structure that we can exploit further. They are defined entirely out of exponentiated tensors of Pauli matrices, giving the regular structure described by the `PauliExpBox`es. Under many circumstances, it is more efficient to not synthesise these constructions individually, but simultaneously in groups. The `PauliSimp` pass finds the description of a given circuit as a sequence of `PauliExpBox`es and resynthesises them (by default, in groups of commuting terms). This can cause great change in the overall structure and shape of the circuit, enabling the identification and elimination of non-local redundancy."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.passes import PauliSimp"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_circuit = ucc(arg_values)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count before\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth before\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["PauliSimp().apply(test_circuit)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count after PS\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth after PS\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["FullPeepholeOptimise().apply(test_circuit)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count after PS+FPO\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth after PS+FPO\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["To include this into our routines, we can just add the simplification passes to the objective function. The `get_operator_expectation_value` utility handles compiling to meet the requirements of the backend, so we don't have to worry about that here."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function with circuit simplification:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc(params)\n", " PauliSimp().apply(circ)\n", " FullPeepholeOptimise().apply(circ)\n", " return (\n", " get_operator_expectation_value(circ, hamiltonian_op, backend, n_shots=4000)\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These circuit simplification techniques have tried to preserve the exact unitary of the circuit, but there are ways to change the unitary whilst preserving the correctness of the algorithm as a whole.
\n", "
\n", "For example, the excitation terms are generated by trotterisation of the excitation operator, and the order of the terms does not change the unitary in the limit of many trotter steps, so in this sense we are free to sequence the terms how we like and it is sensible to do this in a way that enables efficient synthesis of the circuit. Prioritising collecting terms into commuting sets is a very beneficial heuristic for this and can be performed using the `gen_term_sequence_circuit` method to group the terms together into collections of `PauliExpBox`es and the `GuidedPauliSimp` pass to utilise these sets for synthesis."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.passes import GuidedPauliSimp\n", "from pytket.utils import gen_term_sequence_circuit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def ucc(params):\n", " singles_params = {qps: params[0] * coeff for qps, coeff in singles.items()}\n", " doubles_params = {qps: params[1] * coeff for qps, coeff in doubles.items()}\n", " excitation_op = QubitPauliOperator({**singles_params, **doubles_params})\n", " reference_circ = Circuit(4).X(1).X(3)\n", " ansatz = gen_term_sequence_circuit(excitation_op, reference_circ)\n", " GuidedPauliSimp().apply(ansatz)\n", " FullPeepholeOptimise().apply(ansatz)\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Adding these simplification routines doesn't come for free. Compiling and simplifying the circuit to achieve the best results possible can be a difficult task, which can take some time for the classical computer to perform.
\n", "
\n", "During a VQE run, we will call this objective function many times and run many measurement circuits within each, but the circuits that are run on the quantum computer are almost identical, having the same gate structure but with different gate parameters and measurements. We have already exploited this within the body of the objective function by simplifying the ansatz circuit before we call `get_operator_expectation_value`, so it is only done once per objective calculation rather than once per measurement circuit.
\n", "
\n", "We can go even further by simplifying it once outside of the objective function, and then instantiating the simplified ansatz with the parameter values needed. For this, we will construct the UCC ansatz circuit using symbolic (parametric) gates."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from sympy import symbols"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Symbolic UCC ansatz generation:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["syms = symbols(\"p0 p1 p2\")\n", "singles_a_syms = {qps: syms[0] * coeff for qps, coeff in singles_a.items()}\n", "singles_b_syms = {qps: syms[1] * coeff for qps, coeff in singles_b.items()}\n", "doubles_syms = {qps: syms[2] * coeff for qps, coeff in doubles.items()}\n", "excitation_op = QubitPauliOperator({**singles_a_syms, **singles_b_syms, **doubles_syms})\n", "ucc_ref = Circuit(4).X(1).X(3)\n", "ucc = gen_term_sequence_circuit(excitation_op, ucc_ref)\n", "GuidedPauliSimp().apply(ucc)\n", "FullPeepholeOptimise().apply(ucc)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function using the symbolic ansatz:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc.copy()\n", " sym_map = dict(zip(syms, params))\n", " circ.symbol_substitution(sym_map)\n", " return (\n", " get_operator_expectation_value(circ, hamiltonian_op, backend, n_shots=4000)\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We have now got some very good use of `pytket` for simplifying each individual circuit used in our experiment and for minimising the amount of time spent compiling, but there is still more we can do in terms of reducing the amount of work the quantum computer has to do. Currently, each (non-trivial) term in our measurement hamiltonian is measured by a different circuit within each expectation value calculation. Measurement reduction techniques exist for identifying when these observables commute and hence can be simultaneously measured, reducing the number of circuits required for the full expectation value calculation.
\n", "
\n", "This is built in to the `get_operator_expectation_value` method and can be applied by specifying a way to partition the measuremrnt terms. `PauliPartitionStrat.CommutingSets` can greatly reduce the number of measurement circuits by combining any number of terms that mutually commute. However, this involves potentially adding an arbitrary Clifford circuit to change the basis of the measurements which can be costly on NISQ devices, so `PauliPartitionStrat.NonConflictingSets` trades off some of the reduction in circuit number to guarantee that only single-qubit gates are introduced."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.partition import PauliPartitionStrat"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function using measurement reduction:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc.copy()\n", " sym_map = dict(zip(syms, params))\n", " circ.symbol_substitution(sym_map)\n", " return (\n", " get_operator_expectation_value(\n", " circ,\n", " operator,\n", " backend,\n", " n_shots=4000,\n", " partition_strat=PauliPartitionStrat.CommutingSets,\n", " )\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "markdown", "metadata": {}, "source": ["At this point, we have completely transformed how our VQE objective function works, improving its resilience to noise, cutting the number of circuits run, and maintaining fast runtimes. In doing this, we have explored a number of the features `pytket` offers that are beneficial to VQE and the UCC method:
\n", "- high-level syntactic constructs for evolution operators;
\n", "- utility methods for easy expectation value calculations;
\n", "- both generic and domain-specific circuit simplification methods;
\n", "- symbolic circuit compilation;
\n", "- measurement reduction for expectation value calculations."]}, {"cell_type": "markdown", "metadata": {}, "source": ["For the sake of completeness, the following gives the full code for the final solution, including passing the objective function to a classical optimiser to find the ground state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import openfermion as of\n", "from scipy.optimize import minimize\n", "from sympy import symbols"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import AerBackend\n", "from pytket.circuit import Circuit, Qubit\n", "from pytket.partition import PauliPartitionStrat\n", "from pytket.passes import GuidedPauliSimp, FullPeepholeOptimise\n", "from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.utils import get_operator_expectation_value, gen_term_sequence_circuit\n", "from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Obtain electronic Hamiltonian:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian = (\n", " -0.8153001706270075 * of.QubitOperator(\"\")\n", " + 0.16988452027940318 * of.QubitOperator(\"Z0\")\n", " + -0.21886306781219608 * of.QubitOperator(\"Z1\")\n", " + 0.16988452027940323 * of.QubitOperator(\"Z2\")\n", " + -0.2188630678121961 * of.QubitOperator(\"Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z0 Z1\")\n", " + 0.16821198673715723 * of.QubitOperator(\"Z0 Z2\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z0 Z3\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z1 Z2\")\n", " + 0.1739537877649417 * of.QubitOperator(\"Z1 Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z2 Z3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 Y2 Y3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 Y2 Y3\")\n", ")\n", "nuclear_repulsion_energy = 0.70556961456"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["pauli_sym = {\"I\": Pauli.I, \"X\": Pauli.X, \"Y\": Pauli.Y, \"Z\": Pauli.Z}"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qps_from_openfermion(paulis):\n", " \"\"\"Convert OpenFermion tensor of Paulis to pytket QubitPauliString.\"\"\"\n", " qlist = []\n", " plist = []\n", " for q, p in paulis:\n", " qlist.append(Qubit(q))\n", " plist.append(pauli_sym[p])\n", " return QubitPauliString(qlist, plist)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qpo_from_openfermion(openf_op):\n", " \"\"\"Convert OpenFermion QubitOperator to pytket QubitPauliOperator.\"\"\"\n", " tk_op = dict()\n", " for term, coeff in openf_op.terms.items():\n", " string = qps_from_openfermion(term)\n", " tk_op[string] = coeff\n", " return QubitPauliOperator(tk_op)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian_op = qpo_from_openfermion(hamiltonian)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Obtain terms for single and double excitations:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["q = [Qubit(i) for i in range(4)]\n", "xyii = QubitPauliString([q[0], q[1]], [Pauli.X, Pauli.Y])\n", "yxii = QubitPauliString([q[0], q[1]], [Pauli.Y, Pauli.X])\n", "iixy = QubitPauliString([q[2], q[3]], [Pauli.X, Pauli.Y])\n", "iiyx = QubitPauliString([q[2], q[3]], [Pauli.Y, Pauli.X])\n", "xxxy = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.X, Pauli.Y])\n", "xxyx = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.Y, Pauli.X])\n", "xyxx = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.X, Pauli.X])\n", "yxxx = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.X, Pauli.X])\n", "yyyx = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.Y, Pauli.X])\n", "yyxy = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.X, Pauli.Y])\n", "yxyy = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.Y, Pauli.Y])\n", "xyyy = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.Y, Pauli.Y])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Symbolic UCC ansatz generation:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["syms = symbols(\"p0 p1 p2\")\n", "singles_syms = {xyii: syms[0], yxii: -syms[0], iixy: syms[1], iiyx: -syms[1]}\n", "doubles_syms = {\n", " xxxy: 0.25 * syms[2],\n", " xxyx: -0.25 * syms[2],\n", " xyxx: 0.25 * syms[2],\n", " yxxx: -0.25 * syms[2],\n", " yyyx: -0.25 * syms[2],\n", " yyxy: 0.25 * syms[2],\n", " yxyy: -0.25 * syms[2],\n", " xyyy: 0.25 * syms[2],\n", "}\n", "excitation_op = QubitPauliOperator({**singles_syms, **doubles_syms})\n", "ucc_ref = Circuit(4).X(0).X(2)\n", "ucc = gen_term_sequence_circuit(excitation_op, ucc_ref)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Circuit simplification:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["GuidedPauliSimp().apply(ucc)\n", "FullPeepholeOptimise().apply(ucc)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Connect to a simulator/device:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc.copy()\n", " sym_map = dict(zip(syms, params))\n", " circ.symbol_substitution(sym_map)\n", " return (\n", " get_operator_expectation_value(\n", " circ,\n", " hamiltonian_op,\n", " backend,\n", " n_shots=4000,\n", " partition_strat=PauliPartitionStrat.CommutingSets,\n", " )\n", " + nuclear_repulsion_energy\n", " ).real"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Optimise against the objective function:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["initial_params = [1e-4, 1e-4, 4e-1]\n", "# #result = minimize(objective, initial_params, method=\"Nelder-Mead\")\n", "# #print(\"Final parameter values\", result.x)\n", "# #print(\"Final energy value\", result.fun)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Exercises:
\n", "- Replace the `get_operator_expectation_value` call with its implementation and use this to pull the analysis for measurement reduction outside of the objective function, so our circuits can be fully determined and compiled once. This means that the `symbol_substitution` method will need to be applied to each measurement circuit instead of just the state preparation circuit.
\n", "- Use the `SpamCorrecter` class to add some mitigation of the measurement errors. Start by running the characterisation circuits first, before your main VQE loop, then apply the mitigation to each of the circuits run within the objective function.
\n", "- Change the `backend` by passing in a `Qiskit` `NoiseModel` to simulate a noisy device. Compare the accuracy of the objective function both with and without the circuit simplification. Try running a classical optimiser over the objective function and compare the convergence rates with different noise models. If you have access to a QPU, try changing the `backend` to connect to that and compare the results to the simulator."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file +{"cells": [{"cell_type": "markdown", "metadata": {}, "source": ["# VQE with UCC ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["In this tutorial, we will focus on:
\n", "- building parameterised ans\u00e4tze for variational algorithms;
\n", "- compilation tools for UCC-style ans\u00e4tze."]}, {"cell_type": "markdown", "metadata": {}, "source": ["This example assumes the reader is familiar with the Variational Quantum Eigensolver and its application to electronic structure problems through the Unitary Coupled Cluster approach.
\n", "
\n", "To run this example, you will need `pytket` and `pytket-qiskit`, as well as `openfermion`, `scipy`, and `sympy`.
\n", "
\n", "We will start with a basic implementation and then gradually modify it to make it faster, more general, and less noisy. The final solution is given in full at the bottom of the notebook.
\n", "
\n", "Suppose we have some electronic configuration problem, expressed via a physical Hamiltonian. (The Hamiltonian and excitations in this example were obtained using `qiskit-aqua` version 0.5.2 and `pyscf` for H2, bond length 0.75A, sto3g basis, Jordan-Wigner encoding, with no qubit reduction or orbital freezing.). We express it succinctly using the openfermion library:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import openfermion as of"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian = (\n", " -0.8153001706270075 * of.QubitOperator(\"\")\n", " + 0.16988452027940318 * of.QubitOperator(\"Z0\")\n", " + -0.21886306781219608 * of.QubitOperator(\"Z1\")\n", " + 0.16988452027940323 * of.QubitOperator(\"Z2\")\n", " + -0.2188630678121961 * of.QubitOperator(\"Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z0 Z1\")\n", " + 0.16821198673715723 * of.QubitOperator(\"Z0 Z2\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z0 Z3\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z1 Z2\")\n", " + 0.1739537877649417 * of.QubitOperator(\"Z1 Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z2 Z3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 Y2 Y3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 Y2 Y3\")\n", ")\n", "nuclear_repulsion_energy = 0.70556961456"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We would like to define our ansatz for arbitrary parameter values. For simplicity, let's start with a Hardware Efficient Ansatz."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import Circuit"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Hardware efficient ansatz:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def hea(params):\n", " ansatz = Circuit(4)\n", " for i in range(4):\n", " ansatz.Ry(params[i], i)\n", " for i in range(3):\n", " ansatz.CX(i, i + 1)\n", " for i in range(4):\n", " ansatz.Ry(params[4 + i], i)\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We can use this to build the objective function for our optimisation."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import AerBackend\n", "from pytket.utils.expectations import expectation_from_counts"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Naive objective function:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " energy = 0\n", " for term, coeff in hamiltonian.terms.items():\n", " if not term:\n", " energy += coeff\n", " continue\n", " circ = hea(params)\n", " circ.add_c_register(\"c\", len(term))\n", " for i, (q, pauli) in enumerate(term):\n", " if pauli == \"X\":\n", " circ.H(q)\n", " elif pauli == \"Y\":\n", " circ.V(q)\n", " circ.Measure(q, i)\n", " compiled_circ = backend.get_compiled_circuit(circ)\n", " counts = backend.run_circuit(compiled_circ, n_shots=4000).get_counts()\n", " energy += coeff * expectation_from_counts(counts)\n", " return energy + nuclear_repulsion_energy"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This objective function is then run through a classical optimiser to find the set of parameter values that minimise the energy of the system. For the sake of example, we will just run this with a single parameter value."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["arg_values = [\n", " -7.31158201e-02,\n", " -1.64514836e-04,\n", " 1.12585591e-03,\n", " -2.58367544e-03,\n", " 1.00006068e00,\n", " -1.19551357e-03,\n", " 9.99963988e-01,\n", " 2.53283285e-03,\n", "]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["energy = objective(arg_values)\n", "print(energy)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The HEA is designed to cram as many orthogonal degrees of freedom into a small circuit as possible to be able to explore a large region of the Hilbert space whilst the circuits themselves can be run with minimal noise. These ans\u00e4tze give virtually-optimal circuits by design, but suffer from an excessive number of variational parameters making convergence slow, barren plateaus where the classical optimiser fails to make progress, and spanning a space where most states lack a physical interpretation. These drawbacks can necessitate adding penalties and may mean that the ansatz cannot actually express the true ground state.
\n", "
\n", "The UCC ansatz, on the other hand, is derived from the electronic configuration. It sacrifices efficiency of the circuit for the guarantee of physical states and the variational parameters all having some meaningful effect, which helps the classical optimisation to converge.
\n", "
\n", "This starts by defining the terms of our single and double excitations. These would usually be generated using the orbital configurations, so we will just use a hard-coded example here for the purposes of demonstration."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import Qubit\n", "from pytket.pauli import Pauli, QubitPauliString"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["q = [Qubit(i) for i in range(4)]\n", "xyii = QubitPauliString([q[0], q[1]], [Pauli.X, Pauli.Y])\n", "yxii = QubitPauliString([q[0], q[1]], [Pauli.Y, Pauli.X])\n", "iixy = QubitPauliString([q[2], q[3]], [Pauli.X, Pauli.Y])\n", "iiyx = QubitPauliString([q[2], q[3]], [Pauli.Y, Pauli.X])\n", "xxxy = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.X, Pauli.Y])\n", "xxyx = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.Y, Pauli.X])\n", "xyxx = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.X, Pauli.X])\n", "yxxx = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.X, Pauli.X])\n", "yyyx = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.Y, Pauli.X])\n", "yyxy = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.X, Pauli.Y])\n", "yxyy = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.Y, Pauli.Y])\n", "xyyy = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.Y, Pauli.Y])"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["singles_a = {xyii: 1.0, yxii: -1.0}\n", "singles_b = {iixy: 1.0, iiyx: -1.0}\n", "doubles = {\n", " xxxy: 0.25,\n", " xxyx: -0.25,\n", " xyxx: 0.25,\n", " yxxx: -0.25,\n", " yyyx: -0.25,\n", " yyxy: 0.25,\n", " yxyy: -0.25,\n", " xyyy: 0.25,\n", "}"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Building the ansatz circuit itself is often done naively by defining the map from each term down to basic gates and then applying it to each term."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def add_operator_term(circuit: Circuit, term: QubitPauliString, angle: float):\n", " qubits = []\n", " for q, p in term.map.items():\n", " if p != Pauli.I:\n", " qubits.append(q)\n", " if p == Pauli.X:\n", " circuit.H(q)\n", " elif p == Pauli.Y:\n", " circuit.V(q)\n", " for i in range(len(qubits) - 1):\n", " circuit.CX(i, i + 1)\n", " circuit.Rz(angle, len(qubits) - 1)\n", " for i in reversed(range(len(qubits) - 1)):\n", " circuit.CX(i, i + 1)\n", " for q, p in term.map.items():\n", " if p == Pauli.X:\n", " circuit.H(q)\n", " elif p == Pauli.Y:\n", " circuit.Vdg(q)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Unitary Coupled Cluster Singles & Doubles ansatz:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def ucc(params):\n", " ansatz = Circuit(4)\n", " # Set initial reference state\n", " ansatz.X(1).X(3)\n", " # Evolve by excitations\n", " for term, coeff in singles_a.items():\n", " add_operator_term(ansatz, term, coeff * params[0])\n", " for term, coeff in singles_b.items():\n", " add_operator_term(ansatz, term, coeff * params[1])\n", " for term, coeff in doubles.items():\n", " add_operator_term(ansatz, term, coeff * params[2])\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is already quite verbose, but `pytket` has a neat shorthand construction for these operator terms using the `PauliExpBox` construction. We can then decompose these into basic gates using the `DecomposeBoxes` compiler pass."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.circuit import PauliExpBox\n", "from pytket.passes import DecomposeBoxes"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def add_excitation(circ, term_dict, param):\n", " for term, coeff in term_dict.items():\n", " qubits, paulis = zip(*term.map.items())\n", " pbox = PauliExpBox(paulis, coeff * param)\n", " circ.add_pauliexpbox(pbox, qubits)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["UCC ansatz with syntactic shortcuts:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def ucc(params):\n", " ansatz = Circuit(4)\n", " ansatz.X(1).X(3)\n", " add_excitation(ansatz, singles_a, params[0])\n", " add_excitation(ansatz, singles_b, params[1])\n", " add_excitation(ansatz, doubles, params[2])\n", " DecomposeBoxes().apply(ansatz)\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["The objective function can also be simplified using a utility method for constructing the measurement circuits and processing for expectation value calculations. For that, we convert the Hamiltonian to a pytket QubitPauliOperator:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["pauli_sym = {\"I\": Pauli.I, \"X\": Pauli.X, \"Y\": Pauli.Y, \"Z\": Pauli.Z}"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qps_from_openfermion(paulis):\n", " \"\"\"Convert OpenFermion tensor of Paulis to pytket QubitPauliString.\"\"\"\n", " qlist = []\n", " plist = []\n", " for q, p in paulis:\n", " qlist.append(Qubit(q))\n", " plist.append(pauli_sym[p])\n", " return QubitPauliString(qlist, plist)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qpo_from_openfermion(openf_op):\n", " \"\"\"Convert OpenFermion QubitOperator to pytket QubitPauliOperator.\"\"\"\n", " tk_op = dict()\n", " for term, coeff in openf_op.terms.items():\n", " string = qps_from_openfermion(term)\n", " tk_op[string] = coeff\n", " return QubitPauliOperator(tk_op)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian_op = qpo_from_openfermion(hamiltonian)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Simplified objective function using utilities:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.utils.expectations import get_operator_expectation_value"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc(params)\n", " return (\n", " get_operator_expectation_value(circ, hamiltonian_op, backend, n_shots=4000)\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["arg_values = [-3.79002933e-05, 2.42964799e-05, 4.63447157e-01]"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["energy = objective(arg_values)\n", "print(energy)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["This is now the simplest form that this operation can take, but it isn't necessarily the most effective. When we decompose the ansatz circuit into basic gates, it is still very expensive. We can employ some of the circuit simplification passes available in `pytket` to reduce its size and improve fidelity in practice.
\n", "
\n", "A good example is to decompose each `PauliExpBox` into basic gates and then apply `FullPeepholeOptimise`, which defines a compilation strategy utilising all of the simplifications in `pytket` that act locally on small regions of a circuit. We can examine the effectiveness by looking at the number of two-qubit gates before and after simplification, which tends to be a good indicator of fidelity for near-term systems where these gates are often slow and inaccurate."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket import OpType\n", "from pytket.passes import FullPeepholeOptimise"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_circuit = ucc(arg_values)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count before\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth before\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["FullPeepholeOptimise().apply(test_circuit)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count after FPO\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth after FPO\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These simplification techniques are very general and are almost always beneficial to apply to a circuit if you want to eliminate local redundancies. But UCC ans\u00e4tze have extra structure that we can exploit further. They are defined entirely out of exponentiated tensors of Pauli matrices, giving the regular structure described by the `PauliExpBox`es. Under many circumstances, it is more efficient to not synthesise these constructions individually, but simultaneously in groups. The `PauliSimp` pass finds the description of a given circuit as a sequence of `PauliExpBox`es and resynthesises them (by default, in groups of commuting terms). This can cause great change in the overall structure and shape of the circuit, enabling the identification and elimination of non-local redundancy."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.passes import PauliSimp"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["test_circuit = ucc(arg_values)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count before\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth before\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["PauliSimp().apply(test_circuit)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count after PS\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth after PS\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["FullPeepholeOptimise().apply(test_circuit)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["print(\"CX count after PS+FPO\", test_circuit.n_gates_of_type(OpType.CX))\n", "print(\"CX depth after PS+FPO\", test_circuit.depth_by_type(OpType.CX))"]}, {"cell_type": "markdown", "metadata": {}, "source": ["To include this into our routines, we can just add the simplification passes to the objective function. The `get_operator_expectation_value` utility handles compiling to meet the requirements of the backend, so we don't have to worry about that here."]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function with circuit simplification:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc(params)\n", " PauliSimp().apply(circ)\n", " FullPeepholeOptimise().apply(circ)\n", " return (\n", " get_operator_expectation_value(circ, hamiltonian_op, backend, n_shots=4000)\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "markdown", "metadata": {}, "source": ["These circuit simplification techniques have tried to preserve the exact unitary of the circuit, but there are ways to change the unitary whilst preserving the correctness of the algorithm as a whole.
\n", "
\n", "For example, the excitation terms are generated by trotterisation of the excitation operator, and the order of the terms does not change the unitary in the limit of many trotter steps, so in this sense we are free to sequence the terms how we like and it is sensible to do this in a way that enables efficient synthesis of the circuit. Prioritising collecting terms into commuting sets is a very beneficial heuristic for this and can be performed using the `gen_term_sequence_circuit` method to group the terms together into collections of `PauliExpBox`es and the `GuidedPauliSimp` pass to utilise these sets for synthesis."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.passes import GuidedPauliSimp\n", "from pytket.utils import gen_term_sequence_circuit"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def ucc(params):\n", " singles_params = {qps: params[0] * coeff for qps, coeff in singles.items()}\n", " doubles_params = {qps: params[1] * coeff for qps, coeff in doubles.items()}\n", " excitation_op = QubitPauliOperator({**singles_params, **doubles_params})\n", " reference_circ = Circuit(4).X(1).X(3)\n", " ansatz = gen_term_sequence_circuit(excitation_op, reference_circ)\n", " GuidedPauliSimp().apply(ansatz)\n", " FullPeepholeOptimise().apply(ansatz)\n", " return ansatz"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Adding these simplification routines doesn't come for free. Compiling and simplifying the circuit to achieve the best results possible can be a difficult task, which can take some time for the classical computer to perform.
\n", "
\n", "During a VQE run, we will call this objective function many times and run many measurement circuits within each, but the circuits that are run on the quantum computer are almost identical, having the same gate structure but with different gate parameters and measurements. We have already exploited this within the body of the objective function by simplifying the ansatz circuit before we call `get_operator_expectation_value`, so it is only done once per objective calculation rather than once per measurement circuit.
\n", "
\n", "We can go even further by simplifying it once outside of the objective function, and then instantiating the simplified ansatz with the parameter values needed. For this, we will construct the UCC ansatz circuit using symbolic (parametric) gates."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from sympy import symbols"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Symbolic UCC ansatz generation:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["syms = symbols(\"p0 p1 p2\")\n", "singles_a_syms = {qps: syms[0] * coeff for qps, coeff in singles_a.items()}\n", "singles_b_syms = {qps: syms[1] * coeff for qps, coeff in singles_b.items()}\n", "doubles_syms = {qps: syms[2] * coeff for qps, coeff in doubles.items()}\n", "excitation_op = QubitPauliOperator({**singles_a_syms, **singles_b_syms, **doubles_syms})\n", "ucc_ref = Circuit(4).X(1).X(3)\n", "ucc = gen_term_sequence_circuit(excitation_op, ucc_ref)\n", "GuidedPauliSimp().apply(ucc)\n", "FullPeepholeOptimise().apply(ucc)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function using the symbolic ansatz:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc.copy()\n", " sym_map = dict(zip(syms, params))\n", " circ.symbol_substitution(sym_map)\n", " return (\n", " get_operator_expectation_value(circ, hamiltonian_op, backend, n_shots=4000)\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "markdown", "metadata": {}, "source": ["We have now got some very good use of `pytket` for simplifying each individual circuit used in our experiment and for minimising the amount of time spent compiling, but there is still more we can do in terms of reducing the amount of work the quantum computer has to do. Currently, each (non-trivial) term in our measurement hamiltonian is measured by a different circuit within each expectation value calculation. Measurement reduction techniques exist for identifying when these observables commute and hence can be simultaneously measured, reducing the number of circuits required for the full expectation value calculation.
\n", "
\n", "This is built in to the `get_operator_expectation_value` method and can be applied by specifying a way to partition the measuremrnt terms. `PauliPartitionStrat.CommutingSets` can greatly reduce the number of measurement circuits by combining any number of terms that mutually commute. However, this involves potentially adding an arbitrary Clifford circuit to change the basis of the measurements which can be costly on NISQ devices, so `PauliPartitionStrat.NonConflictingSets` trades off some of the reduction in circuit number to guarantee that only single-qubit gates are introduced."]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.partition import PauliPartitionStrat"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function using measurement reduction:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc.copy()\n", " sym_map = dict(zip(syms, params))\n", " circ.symbol_substitution(sym_map)\n", " return (\n", " get_operator_expectation_value(\n", " circ,\n", " operator,\n", " backend,\n", " n_shots=4000,\n", " partition_strat=PauliPartitionStrat.CommutingSets,\n", " )\n", " + nuclear_repulsion_energy\n", " )"]}, {"cell_type": "markdown", "metadata": {}, "source": ["At this point, we have completely transformed how our VQE objective function works, improving its resilience to noise, cutting the number of circuits run, and maintaining fast runtimes. In doing this, we have explored a number of the features `pytket` offers that are beneficial to VQE and the UCC method:
\n", "- high-level syntactic constructs for evolution operators;
\n", "- utility methods for easy expectation value calculations;
\n", "- both generic and domain-specific circuit simplification methods;
\n", "- symbolic circuit compilation;
\n", "- measurement reduction for expectation value calculations."]}, {"cell_type": "markdown", "metadata": {}, "source": ["For the sake of completeness, the following gives the full code for the final solution, including passing the objective function to a classical optimiser to find the ground state:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["import openfermion as of\n", "from scipy.optimize import minimize\n", "from sympy import symbols"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["from pytket.extensions.qiskit import AerBackend\n", "from pytket.circuit import Circuit, Qubit\n", "from pytket.partition import PauliPartitionStrat\n", "from pytket.passes import GuidedPauliSimp, FullPeepholeOptimise\n", "from pytket.pauli import Pauli, QubitPauliString\n", "from pytket.utils import get_operator_expectation_value, gen_term_sequence_circuit\n", "from pytket.utils.operators import QubitPauliOperator"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Obtain electronic Hamiltonian:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian = (\n", " -0.8153001706270075 * of.QubitOperator(\"\")\n", " + 0.16988452027940318 * of.QubitOperator(\"Z0\")\n", " + -0.21886306781219608 * of.QubitOperator(\"Z1\")\n", " + 0.16988452027940323 * of.QubitOperator(\"Z2\")\n", " + -0.2188630678121961 * of.QubitOperator(\"Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z0 Z1\")\n", " + 0.16821198673715723 * of.QubitOperator(\"Z0 Z2\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z0 Z3\")\n", " + 0.16549431486978672 * of.QubitOperator(\"Z1 Z2\")\n", " + 0.1739537877649417 * of.QubitOperator(\"Z1 Z3\")\n", " + 0.12005143072546047 * of.QubitOperator(\"Z2 Z3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"X0 X1 Y2 Y3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 X2 X3\")\n", " + 0.04544288414432624 * of.QubitOperator(\"Y0 Y1 Y2 Y3\")\n", ")\n", "nuclear_repulsion_energy = 0.70556961456"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["pauli_sym = {\"I\": Pauli.I, \"X\": Pauli.X, \"Y\": Pauli.Y, \"Z\": Pauli.Z}"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qps_from_openfermion(paulis):\n", " \"\"\"Convert OpenFermion tensor of Paulis to pytket QubitPauliString.\"\"\"\n", " qlist = []\n", " plist = []\n", " for q, p in paulis:\n", " qlist.append(Qubit(q))\n", " plist.append(pauli_sym[p])\n", " return QubitPauliString(qlist, plist)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def qpo_from_openfermion(openf_op):\n", " \"\"\"Convert OpenFermion QubitOperator to pytket QubitPauliOperator.\"\"\"\n", " tk_op = dict()\n", " for term, coeff in openf_op.terms.items():\n", " string = qps_from_openfermion(term)\n", " tk_op[string] = coeff\n", " return QubitPauliOperator(tk_op)"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["hamiltonian_op = qpo_from_openfermion(hamiltonian)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Obtain terms for single and double excitations:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["q = [Qubit(i) for i in range(4)]\n", "xyii = QubitPauliString([q[0], q[1]], [Pauli.X, Pauli.Y])\n", "yxii = QubitPauliString([q[0], q[1]], [Pauli.Y, Pauli.X])\n", "iixy = QubitPauliString([q[2], q[3]], [Pauli.X, Pauli.Y])\n", "iiyx = QubitPauliString([q[2], q[3]], [Pauli.Y, Pauli.X])\n", "xxxy = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.X, Pauli.Y])\n", "xxyx = QubitPauliString(q, [Pauli.X, Pauli.X, Pauli.Y, Pauli.X])\n", "xyxx = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.X, Pauli.X])\n", "yxxx = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.X, Pauli.X])\n", "yyyx = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.Y, Pauli.X])\n", "yyxy = QubitPauliString(q, [Pauli.Y, Pauli.Y, Pauli.X, Pauli.Y])\n", "yxyy = QubitPauliString(q, [Pauli.Y, Pauli.X, Pauli.Y, Pauli.Y])\n", "xyyy = QubitPauliString(q, [Pauli.X, Pauli.Y, Pauli.Y, Pauli.Y])"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Symbolic UCC ansatz generation:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["syms = symbols(\"p0 p1 p2\")\n", "singles_syms = {xyii: syms[0], yxii: -syms[0], iixy: syms[1], iiyx: -syms[1]}\n", "doubles_syms = {\n", " xxxy: 0.25 * syms[2],\n", " xxyx: -0.25 * syms[2],\n", " xyxx: 0.25 * syms[2],\n", " yxxx: -0.25 * syms[2],\n", " yyyx: -0.25 * syms[2],\n", " yyxy: 0.25 * syms[2],\n", " yxyy: -0.25 * syms[2],\n", " xyyy: 0.25 * syms[2],\n", "}\n", "excitation_op = QubitPauliOperator({**singles_syms, **doubles_syms})\n", "ucc_ref = Circuit(4).X(0).X(2)\n", "ucc = gen_term_sequence_circuit(excitation_op, ucc_ref)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Circuit simplification:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["GuidedPauliSimp().apply(ucc)\n", "FullPeepholeOptimise().apply(ucc)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Connect to a simulator/device:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["backend = AerBackend()"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Objective function:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["def objective(params):\n", " circ = ucc.copy()\n", " sym_map = dict(zip(syms, params))\n", " circ.symbol_substitution(sym_map)\n", " return (\n", " get_operator_expectation_value(\n", " circ,\n", " hamiltonian_op,\n", " backend,\n", " n_shots=4000,\n", " partition_strat=PauliPartitionStrat.CommutingSets,\n", " )\n", " + nuclear_repulsion_energy\n", " ).real"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Optimise against the objective function:"]}, {"cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": ["initial_params = [1e-4, 1e-4, 4e-1]\n", "# #result = minimize(objective, initial_params, method=\"Nelder-Mead\")\n", "# #print(\"Final parameter values\", result.x)\n", "# #print(\"Final energy value\", result.fun)"]}, {"cell_type": "markdown", "metadata": {}, "source": ["Exercises:
\n", "- Replace the `get_operator_expectation_value` call with its implementation and use this to pull the analysis for measurement reduction outside of the objective function, so our circuits can be fully determined and compiled once. This means that the `symbol_substitution` method will need to be applied to each measurement circuit instead of just the state preparation circuit.
\n", "- Use the `SpamCorrecter` class to add some mitigation of the measurement errors. Start by running the characterisation circuits first, before your main VQE loop, then apply the mitigation to each of the circuits run within the objective function.
\n", "- Change the `backend` by passing in a `Qiskit` `NoiseModel` to simulate a noisy device. Compare the accuracy of the objective function both with and without the circuit simplification. Try running a classical optimiser over the objective function and compare the convergence rates with different noise models. If you have access to a QPU, try changing the `backend` to connect to that and compare the results to the simulator."]}], "metadata": {"kernelspec": {"display_name": "Python 3", "language": "python", "name": "python3"}, "language_info": {"codemirror_mode": {"name": "ipython", "version": 3}, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.4"}}, "nbformat": 4, "nbformat_minor": 2} \ No newline at end of file From bba990a61e4a2318b5c3bfa6654f6dba88ac4183 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 9 Nov 2023 14:52:14 +0000 Subject: [PATCH 48/51] Ensure VQE notebooks are listed next to one another --- examples/_toc.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/examples/_toc.yml b/examples/_toc.yml index 0c14abee..c26227f1 100644 --- a/examples/_toc.yml +++ b/examples/_toc.yml @@ -28,9 +28,9 @@ parts: chapters: - file: phase_estimation - file: ucc_vqe + - file: pytket-qujax_heisenberg_vqe - file: pytket-qujax-classification - file: pytket-qujax_qaoa - - file: pytket-qujax_heisenberg_vqe - file: expectation_value_example - file: entanglement_swapping - file: spam_example From a889296593c6d7facd9ccc1b85f1ccb2d349c2f7 Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 9 Nov 2023 14:57:40 +0000 Subject: [PATCH 49/51] Fix qujax qaoa and VQE notebooks --- examples/python/pytket-qujax_heisenberg_vqe.py | 2 +- examples/python/pytket-qujax_qaoa.py | 2 +- examples/pytket-qujax_heisenberg_vqe.ipynb | 2 +- examples/pytket-qujax_qaoa.ipynb | 2 +- 4 files changed, 4 insertions(+), 4 deletions(-) diff --git a/examples/python/pytket-qujax_heisenberg_vqe.py b/examples/python/pytket-qujax_heisenberg_vqe.py index 2b367f1d..8f0ba52c 100644 --- a/examples/python/pytket-qujax_heisenberg_vqe.py +++ b/examples/python/pytket-qujax_heisenberg_vqe.py @@ -1,4 +1,4 @@ -# # VQE example with pytket-qujax +# # VQE example with `pytket-qujax` from jax import numpy as jnp, random, value_and_grad, jit from pytket import Circuit diff --git a/examples/python/pytket-qujax_qaoa.py b/examples/python/pytket-qujax_qaoa.py index 980bef50..d077612f 100644 --- a/examples/python/pytket-qujax_qaoa.py +++ b/examples/python/pytket-qujax_qaoa.py @@ -1,4 +1,4 @@ -# # Symbolic circuits with `qujax` and `pytket-qujax` +# # Symbolic circuits with `pytket-qujax` # In this notebook we will show how to manipulate symbolic circuits with the `pytket-qujax` extension. In particular, we will consider a QAOA and an Ising Hamiltonian. from pytket import Circuit diff --git a/examples/pytket-qujax_heisenberg_vqe.ipynb b/examples/pytket-qujax_heisenberg_vqe.ipynb index cc32ff66..c3328b3d 100644 --- a/examples/pytket-qujax_heisenberg_vqe.ipynb +++ b/examples/pytket-qujax_heisenberg_vqe.ipynb @@ -1 +1 @@ -{"cells":[{"cell_type":"markdown","metadata":{},"source":["# VQE example with pytket-qujax"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from jax import numpy as jnp, random, value_and_grad, jit\n","from pytket import Circuit\n","from pytket.circuit.display import render_circuit_jupyter\n","import matplotlib.pyplot as plt"]},{"cell_type":"markdown","metadata":{},"source":["## Let's start with a TKET circuit"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["import qujax\n","from pytket.extensions.qujax.qujax_convert import tk_to_qujax"]},{"cell_type":"markdown","metadata":{},"source":["We place barriers to stop tket automatically rearranging gates and we also store the number of circuit parameters as we'll need this later."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def get_circuit(n_qubits, depth):\n"," n_params = 2 * n_qubits * (depth + 1)\n"," param = jnp.zeros((n_params,))\n"," circuit = Circuit(n_qubits)\n"," k = 0\n"," for i in range(n_qubits):\n"," circuit.H(i)\n"," for i in range(n_qubits):\n"," circuit.Rx(param[k], i)\n"," k += 1\n"," for i in range(n_qubits):\n"," circuit.Ry(param[k], i)\n"," k += 1\n"," for _ in range(depth):\n"," for i in range(0, n_qubits - 1):\n"," circuit.CZ(i, i + 1)\n"," circuit.add_barrier(range(0, n_qubits))\n"," for i in range(n_qubits):\n"," circuit.Rx(param[k], i)\n"," k += 1\n"," for i in range(n_qubits):\n"," circuit.Ry(param[k], i)\n"," k += 1\n"," return circuit, n_params"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_qubits = 4\n","depth = 2\n","circuit, n_params = get_circuit(n_qubits, depth)\n","render_circuit_jupyter(circuit)"]},{"cell_type":"markdown","metadata":{},"source":["## Now let's invoke qujax\n","The `pytket.extensions.qujax.tk_to_qujax` function will generate a parameters -> statetensor function for us."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_st = tk_to_qujax(circuit)"]},{"cell_type":"markdown","metadata":{},"source":["Let's try it out on some random parameters values. Be aware that's JAX's random number generator requires a `jax.random.PRNGkey` every time it's called - more info on that [here](https://jax.readthedocs.io/en/latest/jax.random.html).\n","Be aware that we still have convention where parameters are specified as multiples of $\\pi$ - that is in [0,2]."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["params = random.uniform(random.PRNGKey(0), shape=(n_params,), minval=0.0, maxval=2.0)\n","statetensor = param_to_st(params)\n","print(statetensor)\n","print(statetensor.shape)"]},{"cell_type":"markdown","metadata":{},"source":["Note that this function also has an optional second argument where an initiating `statetensor_in` can be provided. If it is not provided it will default to the all 0s state (as we use here)."]},{"cell_type":"markdown","metadata":{},"source":["We can obtain statevector by simply calling `.flatten()`"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["statevector = statetensor.flatten()\n","statevector.shape"]},{"cell_type":"markdown","metadata":{},"source":["And sampling probabilities by squaring the absolute value of the statevector"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["sample_probs = jnp.square(jnp.abs(statevector))\n","plt.bar(jnp.arange(statevector.size), sample_probs)"]},{"cell_type":"markdown","metadata":{},"source":["## Cost function"]},{"cell_type":"markdown","metadata":{},"source":["Now we have our `param_to_st` function we are free to define a cost function that acts on bitstrings (e.g. maxcut) or integers by directly wrapping a function around `param_to_st`. However, cost functions defined via quantum Hamiltonians are a bit more involved.\n","Fortunately, we can encode an Hamiltonian in JAX via the `qujax.get_statetensor_to_expectation_func` function which generates a statetensor -> expected value function for us.\n","It takes three arguments as input\n","- `gate_seq_seq`: A list of string (or array) lists encoding the gates in each term of the Hamiltonian. I.e. `[['X','X'], ['Y','Y'], ['Z','Z']]` corresponds to $H = aX_iX_j + bY_kY_l + cZ_mZ_n$ with qubit indices $i,j,k,l,m,n$ specified in the second argument and coefficients $a,b,c$ specified in the third argument\n","- `qubit_inds_seq`: A list of integer lists encoding which qubit indices to apply the aforementioned gates. I.e. `[[0, 1],[0,1],[0,1]]`. Must have the same structure as `gate_seq_seq` above.\n","- `coefficients`: A list of floats encoding any coefficients in the Hamiltonian. I.e. `[2.3, 0.8, 1.2]` corresponds to $a=2.3,b=0.8,c=1.2$ above. Must have the same length as the two above arguments."]},{"cell_type":"markdown","metadata":{},"source":["More specifically let's consider the problem of finding the ground state of the quantum Heisenberg Hamiltonian"]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","H = \\sum_{i=1}^{n_\\text{qubits}-1} X_i X_{i+1} + Y_i Y_{i+1} + Z_i Z_{i+1}.\n","\\end{equation}\n","$$\n","\n","As described, we define the Hamiltonian via its gate strings, qubit indices and coefficients."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["hamiltonian_gates = [[\"X\", \"X\"], [\"Y\", \"Y\"], [\"Z\", \"Z\"]] * (n_qubits - 1)\n","hamiltonian_qubit_inds = [\n"," [int(i), int(i) + 1] for i in jnp.repeat(jnp.arange(n_qubits), 3)\n","]\n","coefficients = [1.0] * len(hamiltonian_qubit_inds)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(\"Gates:\\t\", hamiltonian_gates)\n","print(\"Qubits:\\t\", hamiltonian_qubit_inds)\n","print(\"Coefficients:\\t\", coefficients)"]},{"cell_type":"markdown","metadata":{},"source":["Now let's get the Hamiltonian as a pure JAX function"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["st_to_expectation = qujax.get_statetensor_to_expectation_func(\n"," hamiltonian_gates, hamiltonian_qubit_inds, coefficients\n",")"]},{"cell_type":"markdown","metadata":{},"source":["Let's check it works on the statetensor we've already generated."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["expected_val = st_to_expectation(statetensor)\n","expected_val"]},{"cell_type":"markdown","metadata":{},"source":["Now let's wrap the `param_to_st` and `st_to_expectation` together to give us an all in one `param_to_expectation` cost function."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_expectation = lambda param: st_to_expectation(param_to_st(param))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_expectation(params)"]},{"cell_type":"markdown","metadata":{},"source":["Sanity check that a different, randomly generated set of parameters gives us a new expected value."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["new_params = random.uniform(\n"," random.PRNGKey(1), shape=(n_params,), minval=0.0, maxval=2.0\n",")\n","param_to_expectation(new_params)"]},{"cell_type":"markdown","metadata":{},"source":["## Exact gradients within a VQE algorithm\n","The `param_to_expectation` function we created is a pure JAX function and outputs a scalar. This means we can pass it to `jax.grad` (or even better `jax.value_and_grad`)."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["cost_and_grad = value_and_grad(param_to_expectation)"]},{"cell_type":"markdown","metadata":{},"source":["The `cost_and_grad` function returns a tuple with the exact cost value and exact gradient evaluated at the parameters."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["cost_and_grad(params)"]},{"cell_type":"markdown","metadata":{},"source":["## Now we have all the tools we need to design our VQE!\n","We'll just use vanilla gradient descent with a constant stepsize"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def vqe(init_param, n_steps, stepsize):\n"," params = jnp.zeros((n_steps, n_params))\n"," params = params.at[0].set(init_param)\n"," cost_vals = jnp.zeros(n_steps)\n"," cost_vals = cost_vals.at[0].set(param_to_expectation(init_param))\n"," for step in range(1, n_steps):\n"," cost_val, cost_grad = cost_and_grad(params[step - 1])\n"," cost_vals = cost_vals.at[step].set(cost_val)\n"," new_param = params[step - 1] - stepsize * cost_grad\n"," params = params.at[step].set(new_param)\n"," print(\"Iteration:\", step, \"\\tCost:\", cost_val, end=\"\\r\")\n"," print(\"\\n\")\n"," return params, cost_vals"]},{"cell_type":"markdown","metadata":{},"source":["Ok enough talking, let's run (and whilst we're at it we'll time it too)"]},{"cell_type":"markdown","metadata":{},"source":["%time"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["vqe_params, vqe_cost_vals = vqe(params, n_steps=250, stepsize=0.01)"]},{"cell_type":"markdown","metadata":{},"source":["Let's plot the results..."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.plot(vqe_cost_vals)\n","plt.xlabel(\"Iteration\")\n","plt.ylabel(\"Cost\")"]},{"cell_type":"markdown","metadata":{},"source":["Pretty good!"]},{"cell_type":"markdown","metadata":{},"source":["## `jax.jit` speedup\n","One last thing... We can significantly speed up the VQE above via the `jax.jit`. In our current implementation, the expensive `cost_and_grad` function is compiled to [XLA](https://www.tensorflow.org/xla) and then executed at each call. By invoking `jax.jit` we ensure that the function is compiled only once (on the first call) and then simply executed at each future call - this is much faster!"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["cost_and_grad = jit(cost_and_grad)"]},{"cell_type":"markdown","metadata":{},"source":["We'll demonstrate this using the second set of initial parameters we randomly generated (to be sure of no caching)."]},{"cell_type":"markdown","metadata":{},"source":["%time"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["new_vqe_params, new_vqe_cost_vals = vqe(new_params, n_steps=250, stepsize=0.01)"]},{"cell_type":"markdown","metadata":{},"source":["That's some speedup!\n","But let's also plot the training to be sure it converged correctly"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.plot(new_vqe_cost_vals)\n","plt.xlabel(\"Iteration\")\n","plt.ylabel(\"Cost\")"]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} +{"cells":[{"cell_type":"markdown","metadata":{},"source":["# VQE example with `pytket-qujax`"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from jax import numpy as jnp, random, value_and_grad, jit\n","from pytket import Circuit\n","from pytket.circuit.display import render_circuit_jupyter\n","import matplotlib.pyplot as plt"]},{"cell_type":"markdown","metadata":{},"source":["## Let's start with a TKET circuit"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["import qujax\n","from pytket.extensions.qujax.qujax_convert import tk_to_qujax"]},{"cell_type":"markdown","metadata":{},"source":["We place barriers to stop tket automatically rearranging gates and we also store the number of circuit parameters as we'll need this later."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def get_circuit(n_qubits, depth):\n"," n_params = 2 * n_qubits * (depth + 1)\n"," param = jnp.zeros((n_params,))\n"," circuit = Circuit(n_qubits)\n"," k = 0\n"," for i in range(n_qubits):\n"," circuit.H(i)\n"," for i in range(n_qubits):\n"," circuit.Rx(param[k], i)\n"," k += 1\n"," for i in range(n_qubits):\n"," circuit.Ry(param[k], i)\n"," k += 1\n"," for _ in range(depth):\n"," for i in range(0, n_qubits - 1):\n"," circuit.CZ(i, i + 1)\n"," circuit.add_barrier(range(0, n_qubits))\n"," for i in range(n_qubits):\n"," circuit.Rx(param[k], i)\n"," k += 1\n"," for i in range(n_qubits):\n"," circuit.Ry(param[k], i)\n"," k += 1\n"," return circuit, n_params"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_qubits = 4\n","depth = 2\n","circuit, n_params = get_circuit(n_qubits, depth)\n","render_circuit_jupyter(circuit)"]},{"cell_type":"markdown","metadata":{},"source":["## Now let's invoke qujax\n","The `pytket.extensions.qujax.tk_to_qujax` function will generate a parameters -> statetensor function for us."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_st = tk_to_qujax(circuit)"]},{"cell_type":"markdown","metadata":{},"source":["Let's try it out on some random parameters values. Be aware that's JAX's random number generator requires a `jax.random.PRNGkey` every time it's called - more info on that [here](https://jax.readthedocs.io/en/latest/jax.random.html).\n","Be aware that we still have convention where parameters are specified as multiples of $\\pi$ - that is in [0,2]."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["params = random.uniform(random.PRNGKey(0), shape=(n_params,), minval=0.0, maxval=2.0)\n","statetensor = param_to_st(params)\n","print(statetensor)\n","print(statetensor.shape)"]},{"cell_type":"markdown","metadata":{},"source":["Note that this function also has an optional second argument where an initiating `statetensor_in` can be provided. If it is not provided it will default to the all 0s state (as we use here)."]},{"cell_type":"markdown","metadata":{},"source":["We can obtain statevector by simply calling `.flatten()`"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["statevector = statetensor.flatten()\n","statevector.shape"]},{"cell_type":"markdown","metadata":{},"source":["And sampling probabilities by squaring the absolute value of the statevector"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["sample_probs = jnp.square(jnp.abs(statevector))\n","plt.bar(jnp.arange(statevector.size), sample_probs)"]},{"cell_type":"markdown","metadata":{},"source":["## Cost function"]},{"cell_type":"markdown","metadata":{},"source":["Now we have our `param_to_st` function we are free to define a cost function that acts on bitstrings (e.g. maxcut) or integers by directly wrapping a function around `param_to_st`. However, cost functions defined via quantum Hamiltonians are a bit more involved.\n","Fortunately, we can encode an Hamiltonian in JAX via the `qujax.get_statetensor_to_expectation_func` function which generates a statetensor -> expected value function for us.\n","It takes three arguments as input\n","- `gate_seq_seq`: A list of string (or array) lists encoding the gates in each term of the Hamiltonian. I.e. `[['X','X'], ['Y','Y'], ['Z','Z']]` corresponds to $H = aX_iX_j + bY_kY_l + cZ_mZ_n$ with qubit indices $i,j,k,l,m,n$ specified in the second argument and coefficients $a,b,c$ specified in the third argument\n","- `qubit_inds_seq`: A list of integer lists encoding which qubit indices to apply the aforementioned gates. I.e. `[[0, 1],[0,1],[0,1]]`. Must have the same structure as `gate_seq_seq` above.\n","- `coefficients`: A list of floats encoding any coefficients in the Hamiltonian. I.e. `[2.3, 0.8, 1.2]` corresponds to $a=2.3,b=0.8,c=1.2$ above. Must have the same length as the two above arguments."]},{"cell_type":"markdown","metadata":{},"source":["More specifically let's consider the problem of finding the ground state of the quantum Heisenberg Hamiltonian"]},{"cell_type":"markdown","metadata":{},"source":["$$\n","\\begin{equation}\n","H = \\sum_{i=1}^{n_\\text{qubits}-1} X_i X_{i+1} + Y_i Y_{i+1} + Z_i Z_{i+1}.\n","\\end{equation}\n","$$\n","\n","As described, we define the Hamiltonian via its gate strings, qubit indices and coefficients."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["hamiltonian_gates = [[\"X\", \"X\"], [\"Y\", \"Y\"], [\"Z\", \"Z\"]] * (n_qubits - 1)\n","hamiltonian_qubit_inds = [\n"," [int(i), int(i) + 1] for i in jnp.repeat(jnp.arange(n_qubits), 3)\n","]\n","coefficients = [1.0] * len(hamiltonian_qubit_inds)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(\"Gates:\\t\", hamiltonian_gates)\n","print(\"Qubits:\\t\", hamiltonian_qubit_inds)\n","print(\"Coefficients:\\t\", coefficients)"]},{"cell_type":"markdown","metadata":{},"source":["Now let's get the Hamiltonian as a pure JAX function"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["st_to_expectation = qujax.get_statetensor_to_expectation_func(\n"," hamiltonian_gates, hamiltonian_qubit_inds, coefficients\n",")"]},{"cell_type":"markdown","metadata":{},"source":["Let's check it works on the statetensor we've already generated."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["expected_val = st_to_expectation(statetensor)\n","expected_val"]},{"cell_type":"markdown","metadata":{},"source":["Now let's wrap the `param_to_st` and `st_to_expectation` together to give us an all in one `param_to_expectation` cost function."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_expectation = lambda param: st_to_expectation(param_to_st(param))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_expectation(params)"]},{"cell_type":"markdown","metadata":{},"source":["Sanity check that a different, randomly generated set of parameters gives us a new expected value."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["new_params = random.uniform(\n"," random.PRNGKey(1), shape=(n_params,), minval=0.0, maxval=2.0\n",")\n","param_to_expectation(new_params)"]},{"cell_type":"markdown","metadata":{},"source":["## Exact gradients within a VQE algorithm\n","The `param_to_expectation` function we created is a pure JAX function and outputs a scalar. This means we can pass it to `jax.grad` (or even better `jax.value_and_grad`)."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["cost_and_grad = value_and_grad(param_to_expectation)"]},{"cell_type":"markdown","metadata":{},"source":["The `cost_and_grad` function returns a tuple with the exact cost value and exact gradient evaluated at the parameters."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["cost_and_grad(params)"]},{"cell_type":"markdown","metadata":{},"source":["## Now we have all the tools we need to design our VQE!\n","We'll just use vanilla gradient descent with a constant stepsize"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def vqe(init_param, n_steps, stepsize):\n"," params = jnp.zeros((n_steps, n_params))\n"," params = params.at[0].set(init_param)\n"," cost_vals = jnp.zeros(n_steps)\n"," cost_vals = cost_vals.at[0].set(param_to_expectation(init_param))\n"," for step in range(1, n_steps):\n"," cost_val, cost_grad = cost_and_grad(params[step - 1])\n"," cost_vals = cost_vals.at[step].set(cost_val)\n"," new_param = params[step - 1] - stepsize * cost_grad\n"," params = params.at[step].set(new_param)\n"," print(\"Iteration:\", step, \"\\tCost:\", cost_val, end=\"\\r\")\n"," print(\"\\n\")\n"," return params, cost_vals"]},{"cell_type":"markdown","metadata":{},"source":["Ok enough talking, let's run (and whilst we're at it we'll time it too)"]},{"cell_type":"markdown","metadata":{},"source":["%time"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["vqe_params, vqe_cost_vals = vqe(params, n_steps=250, stepsize=0.01)"]},{"cell_type":"markdown","metadata":{},"source":["Let's plot the results..."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.plot(vqe_cost_vals)\n","plt.xlabel(\"Iteration\")\n","plt.ylabel(\"Cost\")"]},{"cell_type":"markdown","metadata":{},"source":["Pretty good!"]},{"cell_type":"markdown","metadata":{},"source":["## `jax.jit` speedup\n","One last thing... We can significantly speed up the VQE above via the `jax.jit`. In our current implementation, the expensive `cost_and_grad` function is compiled to [XLA](https://www.tensorflow.org/xla) and then executed at each call. By invoking `jax.jit` we ensure that the function is compiled only once (on the first call) and then simply executed at each future call - this is much faster!"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["cost_and_grad = jit(cost_and_grad)"]},{"cell_type":"markdown","metadata":{},"source":["We'll demonstrate this using the second set of initial parameters we randomly generated (to be sure of no caching)."]},{"cell_type":"markdown","metadata":{},"source":["%time"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["new_vqe_params, new_vqe_cost_vals = vqe(new_params, n_steps=250, stepsize=0.01)"]},{"cell_type":"markdown","metadata":{},"source":["That's some speedup!\n","But let's also plot the training to be sure it converged correctly"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.plot(new_vqe_cost_vals)\n","plt.xlabel(\"Iteration\")\n","plt.ylabel(\"Cost\")"]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} diff --git a/examples/pytket-qujax_qaoa.ipynb b/examples/pytket-qujax_qaoa.ipynb index b50fd42d..be4466ae 100644 --- a/examples/pytket-qujax_qaoa.ipynb +++ b/examples/pytket-qujax_qaoa.ipynb @@ -1 +1 @@ -{"cells":[{"cell_type":"markdown","metadata":{},"source":["# Symbolic circuits with `qujax` and `pytket-qujax`\n","In this notebook we will show how to manipulate symbolic circuits with the `pytket-qujax` extension. In particular, we will consider a QAOA and an Ising Hamiltonian."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket import Circuit\n","from pytket.circuit.display import render_circuit_jupyter\n","from jax import numpy as jnp, random, value_and_grad, jit\n","from sympy import Symbol\n","import matplotlib.pyplot as plt"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["import qujax\n","from pytket.extensions.qujax import tk_to_qujax"]},{"cell_type":"markdown","metadata":{},"source":["# QAOA\n","The Quantum Approximate Optimization Algorithm (QAOA), first introduced by [Farhi et al.](https://arxiv.org/pdf/1411.4028.pdf), is a quantum variational algorithm used to solve optimization problems. It consists of a unitary $U(\\beta, \\gamma)$ formed by alternate repetitions of $U(\\beta)=e^{-i\\beta H_B}$ and $U(\\gamma)=e^{-i\\gamma H_P}$, where $H_B$ is the mixing Hamiltonian and $H_P$ the problem Hamiltonian. The goal is to find the optimal parameters that minimize $H_P$.\n","Given a depth $d$, the expression of the final unitary is $U(\\beta, \\gamma) = U(\\beta_d)U(\\gamma_d)\\cdots U(\\beta_1)U(\\gamma_1)$. Notice that for each repetition the parameters are different.\n","\n","## Problem Hamiltonian\n","QAOA uses a problem dependent ansatz. Therefore, we first need to know the problem that we want to solve. In this case we will consider an Ising Hamiltonian with only $Z$ interactions. Given a set of pairs (or qubit indices) $E$, the problem Hamiltonian will be:\n","\n","$$\n","\\begin{equation}\n","H_P = \\sum_{(i, j) \\in E}\\alpha_{ij}Z_iZ_j,\n","\\end{equation}\n","$$\n","\n","where $\\alpha_{ij}$ are the coefficients.\n","Let's build our problem Hamiltonian with random coefficients and a set of pairs for a given number of qubits:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_qubits = 4\n","hamiltonian_qubit_inds = [(0, 1), (1, 2), (0, 2), (1, 3)]\n","hamiltonian_gates = [[\"Z\", \"Z\"]] * (len(hamiltonian_qubit_inds))"]},{"cell_type":"markdown","metadata":{},"source":["Notice that in order to use the random package from jax we first need to define a seeded key"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["seed = 13\n","key = random.PRNGKey(seed)\n","coefficients = random.uniform(key, shape=(len(hamiltonian_qubit_inds),))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(\"Gates:\\t\", hamiltonian_gates)\n","print(\"Qubits:\\t\", hamiltonian_qubit_inds)\n","print(\"Coefficients:\\t\", coefficients)"]},{"cell_type":"markdown","metadata":{},"source":["## Variational Circuit\n","Before constructing the circuit, we still need to select the mixing Hamiltonian. In our case, we will be using $X$ gates in each qubit, so $H_B = \\sum_{i=1}^{n}X_i$, where $n$ is the number of qubits. Notice that the unitary $U(\\beta)$, given this mixing Hamiltonian, is an $X$ rotation in each qubit with angle $\\beta$.\n","As for the unitary corresponding to the problem Hamiltonian, $U(\\gamma)$, it has the following form:\n","\n","$$\n","\\begin{equation}\n","U(\\gamma)=\\prod_{(i, j) \\in E}e^{-i\\gamma\\alpha_{ij}Z_i Z_j}\n","\\end{equation}\n","$$\n","\n","The operation $e^{-i\\gamma\\alpha_{ij}Z_iZ_j}$ can be performed using two CNOT gates with qubit $i$ as control and qubit $j$ as target and a $Z$ rotation in qubit $j$ in between them, with angle $\\gamma\\alpha_{ij}$.\n","Finally, the initial state used, in general, with the QAOA is an equal superposition of all the basis states. This can be achieved adding a first layer of Hadamard gates in each qubit at the beginning of the circuit."]},{"cell_type":"markdown","metadata":{},"source":["With all the building blocks, let's construct the symbolic circuit using tket. Notice that in order to define the parameters, we use the ```Symbol``` object from the `sympy` package. More info can be found in this [documentation](https://cqcl.github.io/pytket/manual/manual_circuit.html#symbolic-circuits). In order to later convert the circuit to qujax, we need to return the list of symbolic parameters as well."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def qaoa_circuit(n_qubits, depth):\n"," circuit = Circuit(n_qubits)\n"," p_keys = []\n","\n"," # Initial State\n"," for i in range(n_qubits):\n"," circuit.H(i)\n"," for d in range(depth):\n"," # Hamiltonian unitary\n"," gamma_d = Symbol(f\"γ_{d}\")\n"," for index in range(len(hamiltonian_qubit_inds)):\n"," pair = hamiltonian_qubit_inds[index]\n"," coef = coefficients[index]\n"," circuit.CX(pair[0], pair[1])\n"," circuit.Rz(gamma_d * coef, pair[1])\n"," circuit.CX(pair[0], pair[1])\n"," circuit.add_barrier(range(0, n_qubits))\n"," p_keys.append(gamma_d)\n","\n"," # Mixing unitary\n"," beta_d = Symbol(f\"β_{d}\")\n"," for i in range(n_qubits):\n"," circuit.Rx(beta_d, i)\n"," p_keys.append(beta_d)\n"," return circuit, p_keys"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["depth = 3\n","circuit, keys = qaoa_circuit(n_qubits, depth)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["keys"]},{"cell_type":"markdown","metadata":{},"source":["Let's check the circuit:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["render_circuit_jupyter(circuit)"]},{"cell_type":"markdown","metadata":{},"source":["# Now for `qujax`\n","The `pytket.extensions.qujax.tk_to_qujax` function will generate a parameters -> statetensor function for us. However, in order to convert a symbolic circuit we first need to define the `symbol_map`. This object maps each symbol key to their corresponding index. In our case, since the object `keys` contains the symbols in the correct order, we can simply construct the dictionary as follows:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["symbol_map = {keys[i]: i for i in range(len(keys))}"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["symbol_map"]},{"cell_type":"markdown","metadata":{},"source":["Then, we invoke the `tk_to_qujax` with both the circuit and the symbolic map."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_st = tk_to_qujax(circuit, symbol_map=symbol_map)"]},{"cell_type":"markdown","metadata":{},"source":["And we also construct the expectation map using the problem Hamiltonian via qujax:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["st_to_expectation = qujax.get_statetensor_to_expectation_func(\n"," hamiltonian_gates, hamiltonian_qubit_inds, coefficients\n",")"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_expectation = lambda param: st_to_expectation(param_to_st(param))"]},{"cell_type":"markdown","metadata":{},"source":["# Training process\n","We construct a function that, given a parameter vector, returns the value of the cost function and the gradient.\n","We also `jit` to avoid recompilation, this means that the expensive `cost_and_grad` function is compiled once into a very fast XLA (C++) function which is then executed at each iteration. Alternatively, we could get the same speedup by replacing our `for` loop with `jax.lax.scan`. You can read more about JIT compilation in the [JAX documentation](https://jax.readthedocs.io/en/latest/jax-101/02-jitting.html)."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["cost_and_grad = jit(value_and_grad(param_to_expectation))"]},{"cell_type":"markdown","metadata":{},"source":["For the training process we'll use vanilla gradient descent with a constant stepsize:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["seed = 123\n","key = random.PRNGKey(seed)\n","init_param = random.uniform(key, shape=(len(symbol_map),))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_steps = 150\n","stepsize = 0.01"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param = init_param"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["cost_vals = jnp.zeros(n_steps)\n","cost_vals = cost_vals.at[0].set(param_to_expectation(init_param))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["for step in range(1, n_steps):\n"," cost_val, cost_grad = cost_and_grad(param)\n"," cost_vals = cost_vals.at[step].set(cost_val)\n"," param = param - stepsize * cost_grad\n"," print(\"Iteration:\", step, \"\\tCost:\", cost_val, end=\"\\r\")"]},{"cell_type":"markdown","metadata":{},"source":["Let's visualise the gradient descent"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.plot(cost_vals)\n","plt.xlabel(\"Iteration\")\n","plt.ylabel(\"Cost\")"]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} +{"cells":[{"cell_type":"markdown","metadata":{},"source":["# Symbolic circuits with `pytket-qujax`\n","In this notebook we will show how to manipulate symbolic circuits with the `pytket-qujax` extension. In particular, we will consider a QAOA and an Ising Hamiltonian."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from pytket import Circuit\n","from pytket.circuit.display import render_circuit_jupyter\n","from jax import numpy as jnp, random, value_and_grad, jit\n","from sympy import Symbol\n","import matplotlib.pyplot as plt"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["import qujax\n","from pytket.extensions.qujax import tk_to_qujax"]},{"cell_type":"markdown","metadata":{},"source":["# QAOA\n","The Quantum Approximate Optimization Algorithm (QAOA), first introduced by [Farhi et al.](https://arxiv.org/pdf/1411.4028.pdf), is a quantum variational algorithm used to solve optimization problems. It consists of a unitary $U(\\beta, \\gamma)$ formed by alternate repetitions of $U(\\beta)=e^{-i\\beta H_B}$ and $U(\\gamma)=e^{-i\\gamma H_P}$, where $H_B$ is the mixing Hamiltonian and $H_P$ the problem Hamiltonian. The goal is to find the optimal parameters that minimize $H_P$.\n","Given a depth $d$, the expression of the final unitary is $U(\\beta, \\gamma) = U(\\beta_d)U(\\gamma_d)\\cdots U(\\beta_1)U(\\gamma_1)$. Notice that for each repetition the parameters are different.\n","\n","## Problem Hamiltonian\n","QAOA uses a problem dependent ansatz. Therefore, we first need to know the problem that we want to solve. In this case we will consider an Ising Hamiltonian with only $Z$ interactions. Given a set of pairs (or qubit indices) $E$, the problem Hamiltonian will be:\n","\n","$$\n","\\begin{equation}\n","H_P = \\sum_{(i, j) \\in E}\\alpha_{ij}Z_iZ_j,\n","\\end{equation}\n","$$\n","\n","where $\\alpha_{ij}$ are the coefficients.\n","Let's build our problem Hamiltonian with random coefficients and a set of pairs for a given number of qubits:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_qubits = 4\n","hamiltonian_qubit_inds = [(0, 1), (1, 2), (0, 2), (1, 3)]\n","hamiltonian_gates = [[\"Z\", \"Z\"]] * (len(hamiltonian_qubit_inds))"]},{"cell_type":"markdown","metadata":{},"source":["Notice that in order to use the random package from jax we first need to define a seeded key"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["seed = 13\n","key = random.PRNGKey(seed)\n","coefficients = random.uniform(key, shape=(len(hamiltonian_qubit_inds),))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["print(\"Gates:\\t\", hamiltonian_gates)\n","print(\"Qubits:\\t\", hamiltonian_qubit_inds)\n","print(\"Coefficients:\\t\", coefficients)"]},{"cell_type":"markdown","metadata":{},"source":["## Variational Circuit\n","Before constructing the circuit, we still need to select the mixing Hamiltonian. In our case, we will be using $X$ gates in each qubit, so $H_B = \\sum_{i=1}^{n}X_i$, where $n$ is the number of qubits. Notice that the unitary $U(\\beta)$, given this mixing Hamiltonian, is an $X$ rotation in each qubit with angle $\\beta$.\n","As for the unitary corresponding to the problem Hamiltonian, $U(\\gamma)$, it has the following form:\n","\n","$$\n","\\begin{equation}\n","U(\\gamma)=\\prod_{(i, j) \\in E}e^{-i\\gamma\\alpha_{ij}Z_i Z_j}\n","\\end{equation}\n","$$\n","\n","The operation $e^{-i\\gamma\\alpha_{ij}Z_iZ_j}$ can be performed using two CNOT gates with qubit $i$ as control and qubit $j$ as target and a $Z$ rotation in qubit $j$ in between them, with angle $\\gamma\\alpha_{ij}$.\n","Finally, the initial state used, in general, with the QAOA is an equal superposition of all the basis states. This can be achieved adding a first layer of Hadamard gates in each qubit at the beginning of the circuit."]},{"cell_type":"markdown","metadata":{},"source":["With all the building blocks, let's construct the symbolic circuit using tket. Notice that in order to define the parameters, we use the ```Symbol``` object from the `sympy` package. More info can be found in this [documentation](https://cqcl.github.io/pytket/manual/manual_circuit.html#symbolic-circuits). In order to later convert the circuit to qujax, we need to return the list of symbolic parameters as well."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def qaoa_circuit(n_qubits, depth):\n"," circuit = Circuit(n_qubits)\n"," p_keys = []\n","\n"," # Initial State\n"," for i in range(n_qubits):\n"," circuit.H(i)\n"," for d in range(depth):\n"," # Hamiltonian unitary\n"," gamma_d = Symbol(f\"γ_{d}\")\n"," for index in range(len(hamiltonian_qubit_inds)):\n"," pair = hamiltonian_qubit_inds[index]\n"," coef = coefficients[index]\n"," circuit.CX(pair[0], pair[1])\n"," circuit.Rz(gamma_d * coef, pair[1])\n"," circuit.CX(pair[0], pair[1])\n"," circuit.add_barrier(range(0, n_qubits))\n"," p_keys.append(gamma_d)\n","\n"," # Mixing unitary\n"," beta_d = Symbol(f\"β_{d}\")\n"," for i in range(n_qubits):\n"," circuit.Rx(beta_d, i)\n"," p_keys.append(beta_d)\n"," return circuit, p_keys"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["depth = 3\n","circuit, keys = qaoa_circuit(n_qubits, depth)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["keys"]},{"cell_type":"markdown","metadata":{},"source":["Let's check the circuit:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["render_circuit_jupyter(circuit)"]},{"cell_type":"markdown","metadata":{},"source":["# Now for `qujax`\n","The `pytket.extensions.qujax.tk_to_qujax` function will generate a parameters -> statetensor function for us. However, in order to convert a symbolic circuit we first need to define the `symbol_map`. This object maps each symbol key to their corresponding index. In our case, since the object `keys` contains the symbols in the correct order, we can simply construct the dictionary as follows:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["symbol_map = {keys[i]: i for i in range(len(keys))}"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["symbol_map"]},{"cell_type":"markdown","metadata":{},"source":["Then, we invoke the `tk_to_qujax` with both the circuit and the symbolic map."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_st = tk_to_qujax(circuit, symbol_map=symbol_map)"]},{"cell_type":"markdown","metadata":{},"source":["And we also construct the expectation map using the problem Hamiltonian via qujax:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["st_to_expectation = qujax.get_statetensor_to_expectation_func(\n"," hamiltonian_gates, hamiltonian_qubit_inds, coefficients\n",")"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_expectation = lambda param: st_to_expectation(param_to_st(param))"]},{"cell_type":"markdown","metadata":{},"source":["# Training process\n","We construct a function that, given a parameter vector, returns the value of the cost function and the gradient.\n","We also `jit` to avoid recompilation, this means that the expensive `cost_and_grad` function is compiled once into a very fast XLA (C++) function which is then executed at each iteration. Alternatively, we could get the same speedup by replacing our `for` loop with `jax.lax.scan`. You can read more about JIT compilation in the [JAX documentation](https://jax.readthedocs.io/en/latest/jax-101/02-jitting.html)."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["cost_and_grad = jit(value_and_grad(param_to_expectation))"]},{"cell_type":"markdown","metadata":{},"source":["For the training process we'll use vanilla gradient descent with a constant stepsize:"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["seed = 123\n","key = random.PRNGKey(seed)\n","init_param = random.uniform(key, shape=(len(symbol_map),))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_steps = 150\n","stepsize = 0.01"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param = init_param"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["cost_vals = jnp.zeros(n_steps)\n","cost_vals = cost_vals.at[0].set(param_to_expectation(init_param))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["for step in range(1, n_steps):\n"," cost_val, cost_grad = cost_and_grad(param)\n"," cost_vals = cost_vals.at[step].set(cost_val)\n"," param = param - stepsize * cost_grad\n"," print(\"Iteration:\", step, \"\\tCost:\", cost_val, end=\"\\r\")"]},{"cell_type":"markdown","metadata":{},"source":["Let's visualise the gradient descent"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.plot(cost_vals)\n","plt.xlabel(\"Iteration\")\n","plt.ylabel(\"Cost\")"]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} From 4242867400842692a3640587bb25d6500782d0fa Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 9 Nov 2023 14:59:28 +0000 Subject: [PATCH 50/51] fix qujax classification example --- examples/python/pytket-qujax-classification.py | 2 +- examples/pytket-qujax-classification.ipynb | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/examples/python/pytket-qujax-classification.py b/examples/python/pytket-qujax-classification.py index 3106c380..1fe8435c 100644 --- a/examples/python/pytket-qujax-classification.py +++ b/examples/python/pytket-qujax-classification.py @@ -1,4 +1,4 @@ -# # Binary classification using pytket-qujax +# # Binary classification using `pytket-qujax` from jax import numpy as jnp, random, vmap, value_and_grad, jit from pytket import Circuit diff --git a/examples/pytket-qujax-classification.ipynb b/examples/pytket-qujax-classification.ipynb index 78601918..078adb8b 100644 --- a/examples/pytket-qujax-classification.ipynb +++ b/examples/pytket-qujax-classification.ipynb @@ -1 +1 @@ -{"cells":[{"cell_type":"markdown","metadata":{},"source":["# Binary classification using pytket-qujax"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from jax import numpy as jnp, random, vmap, value_and_grad, jit\n","from pytket import Circuit\n","from pytket.circuit.display import render_circuit_jupyter\n","from pytket.extensions.qujax.qujax_convert import tk_to_qujax\n","import matplotlib.pyplot as plt"]},{"cell_type":"markdown","metadata":{},"source":["# Define the classification task\n","We'll try and learn a _donut_ binary classification function (i.e. a bivariate coordinate is labelled 1 if it is inside the donut and 0 if it is outside)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["inner_rad = 0.25\n","outer_rad = 0.75"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def classification_function(x, y):\n"," r = jnp.sqrt(x**2 + y**2)\n"," return jnp.where((r > inner_rad) * (r < outer_rad), 1, 0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["linsp = jnp.linspace(-1, 1, 1000)\n","Z = vmap(lambda x: vmap(lambda y: classification_function(x, y))(linsp))(linsp)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.contourf(linsp, linsp, Z, cmap=\"Purples\")"]},{"cell_type":"markdown","metadata":{},"source":["Now let's generate some data for our quantum circuit to learn from"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_data = 1000\n","x = random.uniform(random.PRNGKey(0), shape=(n_data, 2), minval=-1, maxval=1)\n","y = classification_function(x[:, 0], x[:, 1])"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.scatter(x[:, 0], x[:, 1], alpha=jnp.where(y, 1, 0.2), s=10)"]},{"cell_type":"markdown","metadata":{},"source":["# Quantum circuit time\n","We'll use a variant of data re-uploading [Pérez-Salinas et al](https://doi.org/10.22331/q-2020-02-06-226) to encode the input data, alongside some variational parameters within a quantum circuit classifier"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_qubits = 3\n","depth = 5"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["c = Circuit(n_qubits)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["for layer in range(depth):\n"," for qi in range(n_qubits):\n"," c.Rz(0.0, qi)\n"," c.Ry(0.0, qi)\n"," c.Rz(0.0, qi)\n"," if layer < (depth - 1):\n"," for qi in range(layer, layer + n_qubits - 1, 2):\n"," c.CZ(qi % n_qubits, (qi + 1) % n_qubits)\n"," c.add_barrier(range(n_qubits))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["render_circuit_jupyter(c)"]},{"cell_type":"markdown","metadata":{},"source":["We can use `pytket-qujax` to generate our angles-to-statetensor function."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["angles_to_st = tk_to_qujax(c)"]},{"cell_type":"markdown","metadata":{},"source":["We'll parameterise each angle as\n","\n","$$\n","\\begin{equation}\n","\\theta_k = b_k + w_k \\, x_k\n","\\end{equation}\n","$$\n","\n","where $b_k, w_k$ are variational parameters to be learnt and $x_k = x_0$ if $k$ even, $x_k = x_1$ if $k$ odd for a single bivariate input point $(x_0, x_1)$."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_angles = 3 * n_qubits * depth\n","n_params = 2 * n_angles"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def param_and_x_to_angles(param, x_single):\n"," biases = param[:n_angles]\n"," weights = param[n_angles:]\n"," weights_times_data = jnp.where(\n"," jnp.arange(n_angles) % 2 == 0, weights * x_single[0], weights * x_single[1]\n"," )\n"," angles = biases + weights_times_data\n"," return angles"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_and_x_to_st = lambda param, x_single: angles_to_st(\n"," param_and_x_to_angles(param, x_single)\n",")"]},{"cell_type":"markdown","metadata":{},"source":["We'll measure the first qubit only (if its 1 we label _donut_, if its 0 we label _not donut_)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def param_and_x_to_probability(param, x_single):\n"," st = param_and_x_to_st(param, x_single)\n"," all_probs = jnp.square(jnp.abs(st))\n"," first_qubit_probs = jnp.sum(all_probs, axis=range(1, n_qubits))\n"," return first_qubit_probs[1]"]},{"cell_type":"markdown","metadata":{},"source":["For binary classification, the likelihood for our full data set $(x_{1:N}, y_{1:N})$ is\n","\n","$$\n","\\begin{equation}\n","p(y_{1:N} \\mid b, w, x_{1:N}) = \\prod_{i=1}^N p(y_i \\mid b, w, x_i) = \\prod_{i=1}^N (1 - q_{(b,w)}(x_i))^{I[y_i = 0]}q_{(b,w)}(x_i)^{I[y_i = 1]},\n","\\end{equation}\n","$$\n","\n","where $q_{(b, w)}(x)$ is the probability the quantum circuit classifies input $x$ as donut given variational parameter vectors $(b, w)$. This gives log-likelihood\n","\n","$$\n","\\begin{equation}\n"," \\log p(y_{1:N} \\mid b, w, x_{1:N}) = \\sum_{i=1}^N I[y_i = 0] \\log(1 - q_{(b,w)}(x_i)) + I[y_i = 1] \\log q_{(b,w)}(x_i),\n","\\end{equation}\n","$$\n","\n","which we would like to maximise.\n","\n","Unfortunately, the log-likelihood **cannot** be approximated unbiasedly using shots, that is we can approximate $q_{(b,w)}(x_i)$ unbiasedly but not $\\log(q_{(b,w)}(x_i))$.\n","Note that in qujax simulations we can use the statetensor to calculate this exactly, but it is still good to keep in mind loss functions that can also be used with shots from a quantum device.\n","\n","Instead we can minimise an expected distance between shots and data\n","\n","$$\n","\\begin{equation}\n","C(b, w, x, y) = E_{p(y' \\mid q_{(b, w)}(x))}[\\ell(y', y)] = (1 - q_{(b, w)}(x)) \\ell(0, y) + q_{(b, w)}(x)\\ell(1, y),\n","\\end{equation}\n","$$\n","\n","where $y'$ is a shot, $y$ is a data label and $\\ell$ is some distance between bitstrings - here we simply set $\\ell(0, 0) = \\ell(1, 1) = 0$ and $\\ell(0, 1) = \\ell(1, 0) = 1$ (which coincides with the Hamming distance for this binary example).\n","\n"," The full batch cost function is\n","\n","$$\n","\\begin{equation}\n"," C(b, w) = \\frac1N \\sum_{i=1}^N C(b,\\, w,\\, x_i,\\, y_i).\n","\\end{equation}\n","$$"]},{"cell_type":"markdown","metadata":{},"source":["Note that to calculate the cost function we need to evaluate the statetensor for every input point $x_i$. If the dataset becomes too large, we can easily minibatch."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def param_to_cost(param):\n"," donut_probs = vmap(param_and_x_to_probability, in_axes=(None, 0))(param, x)\n"," costs = jnp.where(y, 1 - donut_probs, donut_probs)\n"," return costs.mean()"]},{"cell_type":"markdown","metadata":{},"source":["# Ready to descend some gradients?\n","We'll just use vanilla gradient descent here"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_cost_and_grad = jit(value_and_grad(param_to_cost))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_iter = 1000\n","stepsize = 1e-1\n","param = random.uniform(random.PRNGKey(1), shape=(n_params,), minval=0, maxval=2)\n","costs = jnp.zeros(n_iter)\n","for i in range(n_iter):\n"," cost, grad = param_to_cost_and_grad(param)\n"," costs = costs.at[i].set(cost)\n"," param = param - stepsize * grad\n"," print(i, \"Cost: \", cost, end=\"\\r\")"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.plot(costs)\n","plt.xlabel(\"Iteration\")\n","plt.ylabel(\"Cost\")"]},{"cell_type":"markdown","metadata":{},"source":["# Visualise trained classifier"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["linsp = jnp.linspace(-1, 1, 100)\n","Z = vmap(\n"," lambda a: vmap(lambda b: param_and_x_to_probability(param, jnp.array([a, b])))(\n"," linsp\n"," )\n",")(linsp)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.contourf(linsp, linsp, Z, cmap=\"Purples\", alpha=0.8)\n","circle_linsp = jnp.linspace(0, 2 * jnp.pi, 100)\n","plt.plot(inner_rad * jnp.cos(circle_linsp), inner_rad * jnp.sin(circle_linsp), c=\"red\")\n","plt.plot(outer_rad * jnp.cos(circle_linsp), outer_rad * jnp.sin(circle_linsp), c=\"red\")"]},{"cell_type":"markdown","metadata":{},"source":["Looks good, it has clearly grasped the donut shape. Sincerest apologies if you are now hungry! 🍩"]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} +{"cells":[{"cell_type":"markdown","metadata":{},"source":["# Binary classification using `pytket-qujax`"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["from jax import numpy as jnp, random, vmap, value_and_grad, jit\n","from pytket import Circuit\n","from pytket.circuit.display import render_circuit_jupyter\n","from pytket.extensions.qujax.qujax_convert import tk_to_qujax\n","import matplotlib.pyplot as plt"]},{"cell_type":"markdown","metadata":{},"source":["# Define the classification task\n","We'll try and learn a _donut_ binary classification function (i.e. a bivariate coordinate is labelled 1 if it is inside the donut and 0 if it is outside)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["inner_rad = 0.25\n","outer_rad = 0.75"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def classification_function(x, y):\n"," r = jnp.sqrt(x**2 + y**2)\n"," return jnp.where((r > inner_rad) * (r < outer_rad), 1, 0)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["linsp = jnp.linspace(-1, 1, 1000)\n","Z = vmap(lambda x: vmap(lambda y: classification_function(x, y))(linsp))(linsp)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.contourf(linsp, linsp, Z, cmap=\"Purples\")"]},{"cell_type":"markdown","metadata":{},"source":["Now let's generate some data for our quantum circuit to learn from"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_data = 1000\n","x = random.uniform(random.PRNGKey(0), shape=(n_data, 2), minval=-1, maxval=1)\n","y = classification_function(x[:, 0], x[:, 1])"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.scatter(x[:, 0], x[:, 1], alpha=jnp.where(y, 1, 0.2), s=10)"]},{"cell_type":"markdown","metadata":{},"source":["# Quantum circuit time\n","We'll use a variant of data re-uploading [Pérez-Salinas et al](https://doi.org/10.22331/q-2020-02-06-226) to encode the input data, alongside some variational parameters within a quantum circuit classifier"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_qubits = 3\n","depth = 5"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["c = Circuit(n_qubits)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["for layer in range(depth):\n"," for qi in range(n_qubits):\n"," c.Rz(0.0, qi)\n"," c.Ry(0.0, qi)\n"," c.Rz(0.0, qi)\n"," if layer < (depth - 1):\n"," for qi in range(layer, layer + n_qubits - 1, 2):\n"," c.CZ(qi % n_qubits, (qi + 1) % n_qubits)\n"," c.add_barrier(range(n_qubits))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["render_circuit_jupyter(c)"]},{"cell_type":"markdown","metadata":{},"source":["We can use `pytket-qujax` to generate our angles-to-statetensor function."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["angles_to_st = tk_to_qujax(c)"]},{"cell_type":"markdown","metadata":{},"source":["We'll parameterise each angle as\n","\n","$$\n","\\begin{equation}\n","\\theta_k = b_k + w_k \\, x_k\n","\\end{equation}\n","$$\n","\n","where $b_k, w_k$ are variational parameters to be learnt and $x_k = x_0$ if $k$ even, $x_k = x_1$ if $k$ odd for a single bivariate input point $(x_0, x_1)$."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_angles = 3 * n_qubits * depth\n","n_params = 2 * n_angles"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def param_and_x_to_angles(param, x_single):\n"," biases = param[:n_angles]\n"," weights = param[n_angles:]\n"," weights_times_data = jnp.where(\n"," jnp.arange(n_angles) % 2 == 0, weights * x_single[0], weights * x_single[1]\n"," )\n"," angles = biases + weights_times_data\n"," return angles"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_and_x_to_st = lambda param, x_single: angles_to_st(\n"," param_and_x_to_angles(param, x_single)\n",")"]},{"cell_type":"markdown","metadata":{},"source":["We'll measure the first qubit only (if its 1 we label _donut_, if its 0 we label _not donut_)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def param_and_x_to_probability(param, x_single):\n"," st = param_and_x_to_st(param, x_single)\n"," all_probs = jnp.square(jnp.abs(st))\n"," first_qubit_probs = jnp.sum(all_probs, axis=range(1, n_qubits))\n"," return first_qubit_probs[1]"]},{"cell_type":"markdown","metadata":{},"source":["For binary classification, the likelihood for our full data set $(x_{1:N}, y_{1:N})$ is\n","\n","$$\n","\\begin{equation}\n","p(y_{1:N} \\mid b, w, x_{1:N}) = \\prod_{i=1}^N p(y_i \\mid b, w, x_i) = \\prod_{i=1}^N (1 - q_{(b,w)}(x_i))^{I[y_i = 0]}q_{(b,w)}(x_i)^{I[y_i = 1]},\n","\\end{equation}\n","$$\n","\n","where $q_{(b, w)}(x)$ is the probability the quantum circuit classifies input $x$ as donut given variational parameter vectors $(b, w)$. This gives log-likelihood\n","\n","$$\n","\\begin{equation}\n"," \\log p(y_{1:N} \\mid b, w, x_{1:N}) = \\sum_{i=1}^N I[y_i = 0] \\log(1 - q_{(b,w)}(x_i)) + I[y_i = 1] \\log q_{(b,w)}(x_i),\n","\\end{equation}\n","$$\n","\n","which we would like to maximise.\n","\n","Unfortunately, the log-likelihood **cannot** be approximated unbiasedly using shots, that is we can approximate $q_{(b,w)}(x_i)$ unbiasedly but not $\\log(q_{(b,w)}(x_i))$.\n","Note that in qujax simulations we can use the statetensor to calculate this exactly, but it is still good to keep in mind loss functions that can also be used with shots from a quantum device.\n","\n","Instead we can minimise an expected distance between shots and data\n","\n","$$\n","\\begin{equation}\n","C(b, w, x, y) = E_{p(y' \\mid q_{(b, w)}(x))}[\\ell(y', y)] = (1 - q_{(b, w)}(x)) \\ell(0, y) + q_{(b, w)}(x)\\ell(1, y),\n","\\end{equation}\n","$$\n","\n","where $y'$ is a shot, $y$ is a data label and $\\ell$ is some distance between bitstrings - here we simply set $\\ell(0, 0) = \\ell(1, 1) = 0$ and $\\ell(0, 1) = \\ell(1, 0) = 1$ (which coincides with the Hamming distance for this binary example).\n","\n"," The full batch cost function is\n","\n","$$\n","\\begin{equation}\n"," C(b, w) = \\frac1N \\sum_{i=1}^N C(b,\\, w,\\, x_i,\\, y_i).\n","\\end{equation}\n","$$"]},{"cell_type":"markdown","metadata":{},"source":["Note that to calculate the cost function we need to evaluate the statetensor for every input point $x_i$. If the dataset becomes too large, we can easily minibatch."]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["def param_to_cost(param):\n"," donut_probs = vmap(param_and_x_to_probability, in_axes=(None, 0))(param, x)\n"," costs = jnp.where(y, 1 - donut_probs, donut_probs)\n"," return costs.mean()"]},{"cell_type":"markdown","metadata":{},"source":["# Ready to descend some gradients?\n","We'll just use vanilla gradient descent here"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["param_to_cost_and_grad = jit(value_and_grad(param_to_cost))"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["n_iter = 1000\n","stepsize = 1e-1\n","param = random.uniform(random.PRNGKey(1), shape=(n_params,), minval=0, maxval=2)\n","costs = jnp.zeros(n_iter)\n","for i in range(n_iter):\n"," cost, grad = param_to_cost_and_grad(param)\n"," costs = costs.at[i].set(cost)\n"," param = param - stepsize * grad\n"," print(i, \"Cost: \", cost, end=\"\\r\")"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.plot(costs)\n","plt.xlabel(\"Iteration\")\n","plt.ylabel(\"Cost\")"]},{"cell_type":"markdown","metadata":{},"source":["# Visualise trained classifier"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["linsp = jnp.linspace(-1, 1, 100)\n","Z = vmap(\n"," lambda a: vmap(lambda b: param_and_x_to_probability(param, jnp.array([a, b])))(\n"," linsp\n"," )\n",")(linsp)"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["plt.contourf(linsp, linsp, Z, cmap=\"Purples\", alpha=0.8)\n","circle_linsp = jnp.linspace(0, 2 * jnp.pi, 100)\n","plt.plot(inner_rad * jnp.cos(circle_linsp), inner_rad * jnp.sin(circle_linsp), c=\"red\")\n","plt.plot(outer_rad * jnp.cos(circle_linsp), outer_rad * jnp.sin(circle_linsp), c=\"red\")"]},{"cell_type":"markdown","metadata":{},"source":["Looks good, it has clearly grasped the donut shape. Sincerest apologies if you are now hungry! 🍩"]}],"metadata":{"kernelspec":{"display_name":"Python 3","language":"python","name":"python3"},"language_info":{"codemirror_mode":{"name":"ipython","version":3},"file_extension":".py","mimetype":"text/x-python","name":"python","nbconvert_exporter":"python","pygments_lexer":"ipython3","version":"3.6.4"}},"nbformat":4,"nbformat_minor":2} From da1b92468c2d2d42e204544e1407cf03bbd4e30f Mon Sep 17 00:00:00 2001 From: CalMacCQ <93673602+CalMacCQ@users.noreply.github.com> Date: Thu, 9 Nov 2023 15:06:13 +0000 Subject: [PATCH 51/51] remove uneeded lines from CONTRIBUTING section --- examples/CONTRIBUTING.md | 5 ----- 1 file changed, 5 deletions(-) diff --git a/examples/CONTRIBUTING.md b/examples/CONTRIBUTING.md index fb7b09cc..c7cac86f 100644 --- a/examples/CONTRIBUTING.md +++ b/examples/CONTRIBUTING.md @@ -1,10 +1,5 @@ # Contributing new notebooks -See the pytket examples built with jupyterbook [here](https://tket.quantinuum.com/examples). - - -## Notes for developers - The sources for all these notebooks are the Python scripts in the `python` directory. The notebook files are generated from them with the `gen-nb` script (which requires `p2j`). Do not edit the notebooks directly; instead edit the