diff --git a/README.md b/README.md index c0054cfa..273696a2 100644 --- a/README.md +++ b/README.md @@ -16,6 +16,14 @@ See the respective [Python](https://github.com/CQCL/tket2/blob/main/tket2-py) READMEs for more information. +## Usage + +The rust crate documentation is available at [docs.rs](https://docs.rs/tket2). + +See the [Getting Started][getting-started] notebook for a quick introduction to using `tket2` in Python. + + [getting-started]: https://github.com/CQCL/tket2/blob/main/tket2-py/examples/1-Getting-Started.ipynb + ## Developing TKET2 See [DEVELOPMENT.md][] for instructions on setting up the development environment. diff --git a/tket2-py/README.md b/tket2-py/README.md index 2fffe844..5b798293 100644 --- a/tket2-py/README.md +++ b/tket2-py/README.md @@ -25,6 +25,13 @@ TKET2 can be installed via `pip`. Requires Python >= 3.10. pip install tket2 ``` +## Usage + +See the [Getting Started][getting-started] guide and the other [examples]. + + [getting-started]: https://github.com/CQCL/tket2/blob/main/tket2-py/examples/1-Getting-Started.ipynb + [examples]: https://github.com/CQCL/tket2/blob/main/tket2-py/examples/ + ## Development This package uses [pyo3](https://pyo3.rs/v0.16.4/) and diff --git a/tket2-py/examples/1-Getting-Started.ipynb b/tket2-py/examples/1-Getting-Started.ipynb new file mode 100644 index 00000000..16ace287 --- /dev/null +++ b/tket2-py/examples/1-Getting-Started.ipynb @@ -0,0 +1,669 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "id": "5662ddb4", + "metadata": {}, + "source": [ + "# Getting started with tket2\n", + "\n", + "This demo notebook gives an overview of currently implemented tket2 features.\n", + "\n", + "Be aware that the library is still in development and some features may not be\n", + "fully implemented or may change in the future.\n", + "\n" + ] + }, + { + "cell_type": "markdown", + "id": "87af9fa1", + "metadata": {}, + "source": [ + "## Installation\n", + "\n", + "To install the library, you can use pip:" + ] + }, + { + "cell_type": "code", + "execution_count": 1, + "id": "25ac0737", + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Requirement already satisfied: tket2 in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (0.0.0a1)\n", + "Note: you may need to restart the kernel to use updated packages.\n" + ] + } + ], + "source": [ + "%pip install tket2" + ] + }, + { + "cell_type": "code", + "execution_count": 2, + "id": "0c2a523d", + "metadata": {}, + "outputs": [], + "source": [ + "from tket2.circuit import Tk2Circuit\n", + "import math" + ] + }, + { + "cell_type": "markdown", + "id": "d9d52817", + "metadata": {}, + "source": [ + "Let's configure pretty printing for the circuits, using the mermaid renderer.\n", + "This will render the circuit graphs in `jupyter-lab`, but it is not currently supported when viewing the library in vscode." + ] + }, + { + "cell_type": "code", + "execution_count": 3, + "id": "2896f51f", + "metadata": {}, + "outputs": [], + "source": [ + "from tket2.circuit import render_circuit_mermaid\n", + "\n", + "setattr(\n", + " Tk2Circuit,\n", + " \"_repr_markdown_\",\n", + " lambda self: f\"```mermaid\\n{render_circuit_mermaid(self)}\\n```\"\n", + ")" + ] + }, + { + "cell_type": "markdown", + "id": "d403fe3b", + "metadata": {}, + "source": [ + "# Defining circuits\n", + "\n", + "There are multiple ways for defining circuits in tket2.\n", + "The library provides two limited builders, and it supports importing circuits from `guppy` and `pytket`.\n", + "\n", + "### Using the commands-based builder\n", + "\n", + "The simplest way is to use the commands-based builder interface `CircuitBuild`.\n", + "It supports constructing pure circuits by listing a series of commands applied to specific qubits." + ] + }, + { + "cell_type": "code", + "execution_count": 4, + "id": "71f02038", + "metadata": {}, + "outputs": [ + { + "data": { + "text/markdown": [ + "```mermaid\n", + "graph LR\n", + " subgraph 0 [\"(0) DFG\"]\n", + " direction LR\n", + " 1[\"(1) Input\"]\n", + " 1--\"0:0
qubit\"-->3\n", + " 1--\"1:1
qubit\"-->4\n", + " 2[\"(2) Output\"]\n", + " 3[\"(3) quantum.tket2.H\"]\n", + " 3--\"0:0
qubit\"-->4\n", + " 4[\"(4) quantum.tket2.CX\"]\n", + " 4--\"0:0
qubit\"-->2\n", + " 4--\"1:1
qubit\"-->2\n", + " end\n", + "\n", + "```" + ], + "text/plain": [ + "" + ] + }, + "execution_count": 4, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "from tket2.circuit.build import CircBuild, H, CX\n", + "\n", + "builder = CircBuild(n_qb=2)\n", + "\n", + "builder.extend([ H(0), CX(0, 1)])\n", + "circ = builder.finish()\n", + "circ" + ] + }, + { + "cell_type": "markdown", + "id": "f851efee-a33e-4261-ade8-a78f878e45e2", + "metadata": {}, + "source": [ + "### Using the Dataflow Builder\n", + "\n", + "The Dataflow Builder is more flexible than `CircBuild`. It lets you connect arbitrary inputs and outputs to each operation.\n", + "This way, you can define circuits that read the same boolean multiple times, or allocate qubits dynamically." + ] + }, + { + "cell_type": "code", + "execution_count": 5, + "id": "3e3c2ed0-892b-42d7-b5e8-1bc81970635b", + "metadata": {}, + "outputs": [ + { + "data": { + "text/markdown": [ + "```mermaid\n", + "graph LR\n", + " subgraph 0 [\"(0) DFG\"]\n", + " direction LR\n", + " 1[\"(1) Input\"]\n", + " 1--\"0:0
qubit\"-->4\n", + " 2[\"(2) Output\"]\n", + " 3[\"(3) quantum.tket2.QAlloc\"]\n", + " 3--\"0:1
qubit\"-->5\n", + " 4[\"(4) quantum.tket2.H\"]\n", + " 4--\"0:0
qubit\"-->5\n", + " 5[\"(5) quantum.tket2.CX\"]\n", + " 5--\"0:0
qubit\"-->6\n", + " 5--\"1:0
qubit\"-->7\n", + " 6[\"(6) quantum.tket2.Measure\"]\n", + " 6--\"0:0
qubit\"-->8\n", + " 6--\"1:0
[]+[]\"-->2\n", + " 7[\"(7) quantum.tket2.Measure\"]\n", + " 7--\"0:0
qubit\"-->9\n", + " 7--\"1:1
[]+[]\"-->2\n", + " 8[\"(8) quantum.tket2.QFree\"]\n", + " 9[\"(9) quantum.tket2.QFree\"]\n", + " end\n", + "\n", + "```" + ], + "text/plain": [ + "" + ] + }, + "execution_count": 5, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "from tket2.circuit.build import Dfg, QB_T, BOOL_T\n", + "from tket2.ops import Tk2Op\n", + "\n", + "# Start building DFG with one qubit input and two boolean outputs\n", + "builder = Dfg(input_types=[QB_T], output_types=[BOOL_T, BOOL_T])\n", + "\n", + "# Qubits and booleans are identified by their \"Wires\" in the graph.\n", + "# We can get the wire for the single input qubit.\n", + "[q0] = builder.inputs()\n", + "\n", + "# And allocate a new qubit\n", + "[q1] = builder.add_op(Tk2Op.QAlloc, []).outs(1)\n", + "\n", + "# Each operation returns the new wires it creates.\n", + "[q0] = builder.add_op(Tk2Op.H, [q0]).outs(1)\n", + "q0, q1 = builder.add_op(Tk2Op.CX, [q0, q1]).outs(2)\n", + "\n", + "# Some operations may have different numbers of inputs and outputs.\n", + "[q0, b0] = builder.add_op(Tk2Op.Measure, [q0]).outs(2)\n", + "[q1, b1] = builder.add_op(Tk2Op.Measure, [q1]).outs(2)\n", + "\n", + "# And some may have no outputs at all.\n", + "builder.add_op(Tk2Op.QFree, [q0])\n", + "builder.add_op(Tk2Op.QFree, [q1])\n", + "\n", + "# To get the final circuit, we need to call finish() with the desired output wires.\n", + "circ = builder.finish([b0, b1])\n", + "\n", + "circ" + ] + }, + { + "cell_type": "markdown", + "id": "e3de0d13", + "metadata": {}, + "source": [ + "### Using pytket\n", + "\n", + "We can convert from and to `pytket` circuits." + ] + }, + { + "cell_type": "code", + "execution_count": 6, + "id": "0ffe2aed", + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Requirement already satisfied: pytket in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (1.29.0)\n", + "Requirement already satisfied: sympy~=1.6 in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (from pytket) (1.12.1)\n", + "Requirement already satisfied: numpy<2.0,>=1.21.4 in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (from pytket) (1.26.4)\n", + "Requirement already satisfied: lark~=1.1 in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (from pytket) (1.1.9)\n", + "Requirement already satisfied: scipy~=1.13 in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (from pytket) (1.13.1)\n", + "Requirement already satisfied: networkx>=2.8.8 in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (from pytket) (3.3)\n", + "Requirement already satisfied: graphviz~=0.14 in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (from pytket) (0.20.3)\n", + "Requirement already satisfied: jinja2~=3.0 in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (from pytket) (3.1.4)\n", + "Requirement already satisfied: types-pkg-resources in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (from pytket) (0.1.3)\n", + "Requirement already satisfied: typing-extensions~=4.2 in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (from pytket) (4.12.2)\n", + "Requirement already satisfied: qwasm~=1.0 in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (from pytket) (1.0.1)\n", + "Requirement already satisfied: MarkupSafe>=2.0 in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (from jinja2~=3.0->pytket) (2.1.5)\n", + "Requirement already satisfied: setuptools in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (from qwasm~=1.0->pytket) (70.0.0)\n", + "Requirement already satisfied: mpmath<1.4.0,>=1.1.0 in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (from sympy~=1.6->pytket) (1.3.0)\n", + "Note: you may need to restart the kernel to use updated packages.\n" + ] + } + ], + "source": [ + "%pip install pytket\n", + "\n", + "from pytket.circuit import Circuit as PytketCircuit\n", + "from pytket.circuit.display import render_circuit_jupyter" + ] + }, + { + "cell_type": "code", + "execution_count": 7, + "id": "f47e0f2d", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "\n", + "\n", + "\n", + "\n", + "\n", + "\n", + "
\n", + " \n", + "
\n", + "\n" + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/markdown": [ + "```mermaid\n", + "graph LR\n", + " subgraph 0 [\"(0) FuncDefn\"]\n", + " direction LR\n", + " 1[\"(1) Input\"]\n", + " 1--\"0:0
qubit\"-->3\n", + " 1--\"1:1
qubit\"-->4\n", + " 2[\"(2) Output\"]\n", + " 3[\"(3) quantum.tket2.H\"]\n", + " 3--\"0:0
qubit\"-->4\n", + " 4[\"(4) quantum.tket2.CX\"]\n", + " 4--\"0:0
qubit\"-->2\n", + " 4--\"1:1
qubit\"-->2\n", + " end\n", + "\n", + "```" + ], + "text/plain": [ + "" + ] + }, + "execution_count": 7, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "tk1_circ = PytketCircuit(2).H(0).CX(0, 1)\n", + "render_circuit_jupyter(tk1_circ)\n", + "\n", + "circ = Tk2Circuit(tk1_circ)\n", + "circ" + ] + }, + { + "cell_type": "markdown", + "id": "e5450c5d", + "metadata": {}, + "source": [ + "### Using guppy\n", + "\n", + "Finally, if you have a circuit defined in `guppy` it can be imported directly into a `Tk2Circuit` object." + ] + }, + { + "cell_type": "code", + "execution_count": 8, + "id": "975dbe01", + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Requirement already satisfied: guppylang in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (0.5.2)\n", + "Requirement already satisfied: graphviz<0.21.0,>=0.20.1 in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (from guppylang) (0.20.3)\n", + "Requirement already satisfied: hugr<0.3.0,>=0.2.1 in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (from guppylang) (0.2.1)\n", + "Requirement already satisfied: networkx<4.0.0,>=3.2.1 in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (from guppylang) (3.3)\n", + "Requirement already satisfied: pydantic<3.0.0,>=2.7.0b1 in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (from guppylang) (2.7.3)\n", + "Requirement already satisfied: typing-extensions<5.0.0,>=4.9.0 in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (from guppylang) (4.12.2)\n", + "Requirement already satisfied: annotated-types>=0.4.0 in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (from pydantic<3.0.0,>=2.7.0b1->guppylang) (0.7.0)\n", + "Requirement already satisfied: pydantic-core==2.18.4 in /Users/agustinborgna/src/tket2/.venv/lib/python3.12/site-packages (from pydantic<3.0.0,>=2.7.0b1->guppylang) (2.18.4)\n", + "Note: you may need to restart the kernel to use updated packages.\n" + ] + } + ], + "source": [ + "%pip install guppylang\n", + "\n", + "from guppylang import guppy\n", + "from guppylang.module import GuppyModule\n", + "from guppylang.prelude import quantum\n", + "from guppylang.prelude.builtins import py\n", + "from guppylang.prelude.quantum import measure, phased_x, qubit, rz, zz_max\n", + "\n", + "# We define a utility function to convert a GuppyModule to a Tk2Circuit.\n", + "# This will be included with guppy in the future.\n", + "from utils import guppy_to_circuit # type: ignore" + ] + }, + { + "cell_type": "code", + "execution_count": 9, + "id": "96065d20", + "metadata": {}, + "outputs": [ + { + "data": { + "text/markdown": [ + "```mermaid\n", + "graph LR\n", + " subgraph 1 [\"(1) DFG\"]\n", + " direction LR\n", + " 2[\"(2) Input\"]\n", + " 2--\"0:0
qubit\"-->8\n", + " 2--\"1:0
qubit\"-->16\n", + " 27[\"(27) Output\"]\n", + " 4[\"(4) const:custom:f64(1.5707963267948966)\"]\n", + " 4--\"0:0
float64\"-->5\n", + " 5[\"(5) LoadConstant\"]\n", + " 5--\"0:1
float64\"-->8\n", + " 6[\"(6) const:custom:f64(-1.5707963267948966)\"]\n", + " 6--\"0:0
float64\"-->7\n", + " 7[\"(7) LoadConstant\"]\n", + " 7--\"0:2
float64\"-->8\n", + " 8[\"(8) quantum.tket2.PhasedX\"]\n", + " 8--\"0:0
qubit\"-->11\n", + " 9[\"(9) const:custom:f64(3.141592653589793)\"]\n", + " 9--\"0:0
float64\"-->10\n", + " 10[\"(10) LoadConstant\"]\n", + " 10--\"0:1
float64\"-->11\n", + " 11[\"(11) quantum.tket2.RzF64\"]\n", + " 11--\"0:0
qubit\"-->20\n", + " 12[\"(12) const:custom:f64(1.5707963267948966)\"]\n", + " 12--\"0:0
float64\"-->13\n", + " 13[\"(13) LoadConstant\"]\n", + " 13--\"0:1
float64\"-->16\n", + " 14[\"(14) const:custom:f64(-1.5707963267948966)\"]\n", + " 14--\"0:0
float64\"-->15\n", + " 15[\"(15) LoadConstant\"]\n", + " 15--\"0:2
float64\"-->16\n", + " 16[\"(16) quantum.tket2.PhasedX\"]\n", + " 16--\"0:0
qubit\"-->19\n", + " 17[\"(17) const:custom:f64(3.141592653589793)\"]\n", + " 17--\"0:0
float64\"-->18\n", + " 18[\"(18) LoadConstant\"]\n", + " 18--\"0:1
float64\"-->19\n", + " 19[\"(19) quantum.tket2.RzF64\"]\n", + " 19--\"0:1
qubit\"-->20\n", + " 20[\"(20) quantum.tket2.ZZMax\"]\n", + " 20--\"0:0
qubit\"-->23\n", + " 20--\"1:0
qubit\"-->25\n", + " 23[\"(23) quantum.tket2.Measure\"]\n", + " 23--\"0:0
qubit\"-->24\n", + " 24[\"(24) quantum.tket2.QFree\"]\n", + " 25[\"(25) quantum.tket2.Measure\"]\n", + " 25--\"0:0
qubit\"-->26\n", + " 25--\"1:0
[]+[]\"-->27\n", + " 26[\"(26) quantum.tket2.QFree\"]\n", + " end\n", + "\n", + "```" + ], + "text/plain": [ + "" + ] + }, + "execution_count": 9, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "# Define a guppy module with a quantum function\n", + "\n", + "module = GuppyModule(\"test\")\n", + "module.load(quantum)\n", + "\n", + "@guppy(module)\n", + "def my_func(q0: qubit, q1: qubit) -> bool:\n", + " q0 = phased_x(q0, py(math.pi / 2), py(-math.pi / 2))\n", + " q0 = rz(q0, py(math.pi))\n", + " q1 = phased_x(q1, py(math.pi / 2), py(-math.pi / 2))\n", + " q1 = rz(q1, py(math.pi))\n", + " q0, q1 = zz_max(q0, q1)\n", + " _ = measure(q0)\n", + " return measure(q1)\n", + "\n", + "circ = guppy_to_circuit(my_func)\n", + "circ" + ] + }, + { + "cell_type": "markdown", + "id": "55974636", + "metadata": {}, + "source": [ + "This can be combined with the pytket conversion to obtain a `pytket` circuit from a guppy definition!" + ] + }, + { + "cell_type": "code", + "execution_count": 10, + "id": "77feb50b", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "\n", + "\n", + "\n", + "\n", + "\n", + "\n", + "
\n", + " \n", + "
\n", + "\n" + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + } + ], + "source": [ + "module = GuppyModule(\"test\")\n", + "module.load(quantum)\n", + "\n", + "@guppy(module)\n", + "def my_func(q0: qubit, q1: qubit) -> bool:\n", + " q0 = phased_x(q0, py(math.pi / 2), py(-math.pi / 2))\n", + " q0 = rz(q0, py(math.pi))\n", + " q1 = phased_x(q1, py(math.pi / 2), py(-math.pi / 2))\n", + " q1 = rz(q1, py(math.pi))\n", + " q0, q1 = zz_max(q0, q1)\n", + " _ = measure(q0)\n", + " return measure(q1)\n", + "\n", + "circ = guppy_to_circuit(my_func)\n", + "tk1_circ = circ.to_tket1()\n", + "\n", + "render_circuit_jupyter(tk1_circ)" + ] + } + ], + "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.12.3" + } + }, + "nbformat": 4, + "nbformat_minor": 5 +} diff --git a/tket2-py/examples/2-Rewriting-Circuits.ipynb b/tket2-py/examples/2-Rewriting-Circuits.ipynb new file mode 100644 index 00000000..57fe279c --- /dev/null +++ b/tket2-py/examples/2-Rewriting-Circuits.ipynb @@ -0,0 +1,886 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "# Manually rewriting tket2 circuits\n", + "\n", + "Tket2 includes a rewrite engine based on pattern matching and replacement. This allows for the application of a sequence of rewrite rules to a circuit, which can be used to simplify or optimize the circuit. In this notebook, we will demonstrate how to use this feature to rewrite circuits." + ] + }, + { + "cell_type": "code", + "execution_count": 1, + "metadata": {}, + "outputs": [], + "source": [ + "import itertools\n", + "import copy\n", + "\n", + "from tket2.circuit import Tk2Circuit\n", + "from tket2.circuit.build import CircBuild, H, from_coms, CX, PauliX, PauliY, PauliZ\n", + "from tket2.pattern import Rule, RuleMatcher\n", + "\n", + "from pytket import Circuit as Tk1Circuit\n", + "from pytket.circuit.display import render_circuit_jupyter\n", + "\n", + "from guppylang import guppy\n", + "from guppylang.module import GuppyModule\n", + "from guppylang.prelude import quantum\n", + "from guppylang.prelude.quantum import qubit, cx, rz, zz_phase\n", + "\n", + "# We define a utility function to convert a GuppyModule to a Tk2Circuit.\n", + "# This will be included with guppy in the future.\n", + "from utils import guppy_to_circuit # type: ignore" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "### Applying simple rewrite rules\n", + "\n", + "In the following example, we will define a couple simplification rules and apply them to a circuit. The rules are:\n", + "\n", + "1. A sequence of `CX` - `Rz` - `CX` gates can be replaced by a single `ZZPhase`\n", + "2. A `ZZPhase(0.5)` gate can be replaced with a single `CX` and some single-qubit gates\n", + "\n", + "Let's define these rules first." + ] + }, + { + "cell_type": "code", + "execution_count": 2, + "metadata": {}, + "outputs": [], + "source": [ + "# Note: Most of this boilerplate will be removed in the future.\n", + "\n", + "# First we define the pattern we want to match, as a Guppy function\n", + "module = GuppyModule(\"m\")\n", + "module.load(quantum)\n", + "@guppy(module)\n", + "def cnots_to_zzphase_lhs(q0: qubit, q1: qubit, angle: float) -> tuple[qubit, qubit]:\n", + " q0, q1 = cx(q0, q1)\n", + " q1 = rz(q1, angle)\n", + " q0, q1 = cx(q0, q1)\n", + " return (q0, q1)\n", + "cnots_to_zzphase_lhs = guppy_to_circuit(cnots_to_zzphase_lhs)\n", + "\n", + "# Then we define the replacement circuit\n", + "module = GuppyModule(\"m\")\n", + "module.load(quantum)\n", + "@guppy(module)\n", + "def cnots_to_zzphase_rhs(q0: qubit, q1: qubit, angle: float) -> tuple[qubit, qubit]:\n", + " q0, q1 = zz_phase(q0, q1, angle)\n", + " return (q0, q1)\n", + "cnots_to_zzphase_rhs = guppy_to_circuit(cnots_to_zzphase_rhs)\n", + "\n", + "# define the Rule, and create a RuleMatcher with it\n", + "cnots_to_zzphase = Rule(cnots_to_zzphase_lhs, cnots_to_zzphase_rhs)\n", + "cnots_matcher = RuleMatcher([cnots_to_zzphase])" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "We can now apply the rules to a circuit." + ] + }, + { + "cell_type": "code", + "execution_count": 3, + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Original circuit:\n" + ] + }, + { + "data": { + "text/html": [ + "\n", + "\n", + "\n", + "\n", + "\n", + "\n", + "
\n", + " \n", + "
\n", + "\n" + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Rewritten circuit:\n" + ] + }, + { + "data": { + "text/html": [ + "\n", + "\n", + "\n", + "\n", + "\n", + "\n", + "
\n", + " \n", + "
\n", + "\n" + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + } + ], + "source": [ + "original_circ = Tk1Circuit(3).CX(1, 2).Rz(1/4, 2).CX(1, 2).CX(0, 1).Rz(1/2, 1).CX(0, 1)\n", + "\n", + "print(\"Original circuit:\")\n", + "render_circuit_jupyter(original_circ)\n", + "\n", + "# Match our rewrite rule once, then apply it.\n", + "merged_circ = Tk2Circuit(original_circ)\n", + "while rewrite := cnots_matcher.find_match(merged_circ):\n", + " merged_circ.apply_rewrite(rewrite)\n", + "\n", + "print(\"Rewritten circuit:\")\n", + "render_circuit_jupyter(merged_circ.to_tket1())" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Since the second rewrite rule does not have free parameters, we can define the pattern in pytket." + ] + }, + { + "cell_type": "code", + "execution_count": 4, + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Rewritten circuit:\n" + ] + }, + { + "data": { + "text/html": [ + "\n", + "\n", + "\n", + "\n", + "\n", + "\n", + "
\n", + " \n", + "
\n", + "\n" + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + } + ], + "source": [ + "# For circuits with no free parameters, we can use pytket directly\n", + "zzphase_to_cnot = Rule(\n", + " Tk1Circuit(2).ZZPhase(0.5, 0, 1),\n", + " Tk1Circuit(2).Rx(3.5, 0).Rz(0.5, 1).Rz(1.5, 0).CX(1, 0).Rz(0.5, 0).Rx(0.5, 0).Rz(0.5, 0)\n", + ")\n", + "zzphase_matcher = RuleMatcher([zzphase_to_cnot])\n", + "\n", + "# Apply the matcher to the previous circuit\n", + "final_circ = copy.deepcopy(merged_circ)\n", + "while rewrite := zzphase_matcher.find_match(final_circ):\n", + " final_circ.apply_rewrite(rewrite)\n", + "\n", + "print(\"Rewritten circuit:\")\n", + "render_circuit_jupyter(final_circ.to_tket1())\n" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "The power of `RuleMatcher` gets unlocked when it is defined with multiple rules to apply at once. It can efficiently match millions of rules simultaneously." + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "### Pauli propagation using rewrite rules\n", + "\n", + "In this example we want to propagate Pauli gates through a circuit, using hand-written rules.\n", + "\n", + "Let's start by defining how Pauli gates propagate through CNOT and Hadamard gates.\n", + "For each pair of (pauli, gate) we define a corresponding replacement (gate, pauli).\n" + ] + }, + { + "cell_type": "code", + "execution_count": 5, + "metadata": {}, + "outputs": [], + "source": [ + "# 'from_coms' takes a list of commands and returns a circuit.\n", + "# It's a shorthand for using the 'CircBuild' builder to build non-parametric circuits.\n", + "\n", + "hadamard_rules = [\n", + " Rule(from_coms(PauliX(0), H(0)), from_coms(H(0), PauliZ(0))),\n", + " Rule(from_coms(PauliZ(0), H(0)), from_coms(H(0), PauliX(0))),\n", + "]\n", + "\n", + "cx_rules = [\n", + " Rule(from_coms(PauliZ(0), CX(0, 1)), from_coms(CX(0, 1), PauliZ(0))),\n", + " Rule(from_coms(PauliX(1), CX(0, 1)), from_coms(CX(0, 1), PauliX(1))),\n", + " Rule(from_coms(PauliZ(1), CX(0, 1)), from_coms(CX(0, 1), PauliZ(0), PauliZ(1))),\n", + " Rule(from_coms(PauliX(0), CX(0, 1)), from_coms(CX(0, 1), PauliX(0), PauliX(1))),\n", + "]" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "We can also describe how multiple Pauli gates can be combined with each other." + ] + }, + { + "cell_type": "code", + "execution_count": 6, + "metadata": {}, + "outputs": [], + "source": [ + "def merge_rules() -> list[Rule]:\n", + " paulis = [PauliX(0), PauliY(0), PauliZ(0)]\n", + " identities = [\n", + " Rule(CircBuild(1).extend((p, p)).finish(), CircBuild(1).finish())\n", + " for p in paulis\n", + " ]\n", + "\n", + " off_diag = [\n", + " Rule(\n", + " CircBuild(1).extend((p0, p1)).finish(),\n", + " CircBuild(1).extend((p2,)).finish(),\n", + " )\n", + " for p0, p1, p2 in itertools.permutations(paulis)\n", + " ]\n", + " return [*identities, *off_diag]\n", + "merge_rules = merge_rules()" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "We can now construct a rule matcher that will efficiently find matches for all of these rules in a circuit,\n", + "along with a function to apply all rewrite rules exhaustively." + ] + }, + { + "cell_type": "code", + "execution_count": 7, + "metadata": {}, + "outputs": [], + "source": [ + "matcher = RuleMatcher([*hadamard_rules, *cx_rules, *merge_rules])\n", + "\n", + "def apply_exhaustive(circ: Tk2Circuit, matcher: RuleMatcher) -> int:\n", + " \"\"\"Apply the first matching rule until no more matches are found. Return the\n", + " number of rewrites applied.\"\"\"\n", + " match_count = 0\n", + " while match := matcher.find_match(circ):\n", + " match_count += 1\n", + " circ.apply_rewrite(match)\n", + "\n", + " return match_count" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Finally, we can use our rewriter to a circuit." + ] + }, + { + "cell_type": "code", + "execution_count": 8, + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "\n", + "\n", + "\n", + "\n", + "\n", + "\n", + "
\n", + " \n", + "
\n", + "\n" + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Applied 5 rewrites\n" + ] + }, + { + "data": { + "text/html": [ + "\n", + "\n", + "\n", + "\n", + "\n", + "\n", + "
\n", + " \n", + "
\n", + "\n" + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + } + ], + "source": [ + "circ = CircBuild(2).extend([PauliX(0), PauliZ(1), H(0), CX(0, 1), H(1)]).finish()\n", + "render_circuit_jupyter(circ.to_tket1())\n", + "\n", + "matches = apply_exhaustive(circ, matcher)\n", + "print(f\"Applied {matches} rewrites\")\n", + "\n", + "render_circuit_jupyter(circ.to_tket1())" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "### Supercompilation with Badger\n", + "\n", + "Tket2 includes a compilation pass called Badger which tries to simplify circuits by applying a large number of rewrite rules simultaneously, and searching for the best sequence of rules to apply. This is a computationally expensive process, but can lead to significant simplifications for big circuits." + ] + }, + { + "cell_type": "code", + "execution_count": 9, + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "\n", + "\n", + "\n", + "\n", + "\n", + "\n", + "
\n", + " \n", + "
\n", + "\n" + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "\n", + "\n", + "\n", + "\n", + "\n", + "\n", + "
\n", + " \n", + "
\n", + "\n" + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + } + ], + "source": [ + "from tket2.passes import badger_pass\n", + "\n", + "circ = Tk1Circuit(2).H(1).CZ(0, 1).H(0).H(0).H(1).CX(0, 1)\n", + "render_circuit_jupyter(circ)\n", + "\n", + "badger_pass(rebase=True).apply(circ)\n", + "render_circuit_jupyter(circ)" + ] + } + ], + "metadata": { + "kernelspec": { + "display_name": ".venv", + "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.12.3" + } + }, + "nbformat": 4, + "nbformat_minor": 2 +} diff --git a/tket2-py/examples/utils/__init__.py b/tket2-py/examples/utils/__init__.py new file mode 100644 index 00000000..4da38b54 --- /dev/null +++ b/tket2-py/examples/utils/__init__.py @@ -0,0 +1,21 @@ +"""Some utility functions for the example notebooks.""" + +from tket2.passes import lower_to_pytket +from tket2.circuit import Tk2Circuit + +from guppylang.definition.function import RawFunctionDef + + +# We need to define this helper function for now. It will be included in guppy in the future. +def guppy_to_circuit(func_def: RawFunctionDef) -> Tk2Circuit: + """Convert a Guppy function definition to a `Tk2Circuit`.""" + module = func_def.id.module + assert module is not None, "Function definition must belong to a module" + + hugr = module.compile() + assert hugr is not None, "Module must be compilable" + + json = hugr.to_raw().to_json() + circ = Tk2Circuit.from_guppy_json(json, func_def.name) + + return lower_to_pytket(circ)