microsoft/qdk
Publicmirrored fromhttps://github.com/microsoft/qdkAvailable
samples/qre/0_getting_started.ipynb
1073lines · modecode
| 1 | { |
| 2 | "cells": [ |
| 3 | { |
| 4 | "cell_type": "markdown", |
| 5 | "id": "ab1f991f", |
| 6 | "metadata": {}, |
| 7 | "source": [ |
| 8 | "# Getting started with resource estimation\n", |
| 9 | "\n", |
| 10 | "## High-level overview\n", |
| 11 | "\n", |
| 12 | "The quantum resource estimator takes two inputs: an **application model** that describes the quantum computation (e.g., a Q# program), and an **architecture model** that describes the target quantum hardware (e.g., gate-based superconducting qubits with specific gate times and error rates). From the application, it generates one or more *traces*, compact representations of the instruction sequences applied to qubits. From the architecture, it derives a physical *instruction set* (ISA) specifying each gate's time, qubit cost, and error rate. Between these two layers sit configurable *transforms*: trace transforms such as gate decomposition and layout routines reshape the application trace, while ISA transforms such as quantum error correction codes and magic state factories build higher-fidelity logical instruction sets from the architecture's physical primitives.\n", |
| 13 | "\n", |
| 14 | "<img src=\"overview.svg\" title=\"Resource estimator overview\" style=\"max-width: 1000px; width: 100%\" />\n", |
| 15 | "\n", |
| 16 | "Because there are many valid choices at each layer (rooted in different QEC code distances, different factory protocols, different decomposition parameters), the estimator faces a large combinatorial design space. Rather than evaluating a single fixed configuration, it systematically explores the product of all trace and ISA variants, evaluates physical qubit counts, runtimes, and accumulated error rates for each combination, and then filters the results down to a **Pareto-optimal frontier**: the set of configurations where no other result is simultaneously better in both qubits and runtime while staying within a user-specified error budget.\n", |
| 17 | "\n", |
| 18 | "In practice, a typical estimation call composes these pieces in a single line, like `estimate(app, arch, isa_query=SurfaceCode.q() * RoundBasedFactory.q(), max_error=0.01)`, and returns an `EstimationTable` of Pareto-optimal results that can be inspected, extended with custom columns, or plotted. The remainder of this notebook walks through each step: setting up an application from Q#, choosing an architecture, running the estimation, and interpreting the results." |
| 19 | ] |
| 20 | }, |
| 21 | { |
| 22 | "cell_type": "code", |
| 23 | "execution_count": 1, |
| 24 | "id": "a73b026c", |
| 25 | "metadata": {}, |
| 26 | "outputs": [ |
| 27 | { |
| 28 | "data": { |
| 29 | "application/javascript": "// Copyright (c) Microsoft Corporation.\n// Licensed under the MIT License.\n\n// This file provides CodeMirror syntax highlighting for Q# magic cells\n// in classic Jupyter Notebooks. It does nothing in other (Jupyter Notebook 7,\n// VS Code, Azure Notebooks, etc.) environments.\n\n// Detect the prerequisites and do nothing if they don't exist.\nif (window.require && window.CodeMirror && window.Jupyter) {\n // The simple mode plugin for CodeMirror is not loaded by default, so require it.\n window.require([\"codemirror/addon/mode/simple\"], function defineMode() {\n let rules = [\n {\n token: \"comment\",\n regex: /(\\/\\/).*/,\n beginWord: false,\n },\n {\n token: \"string\",\n regex: String.raw`^\\\"(?:[^\\\"\\\\]|\\\\[\\s\\S])*(?:\\\"|$)`,\n beginWord: false,\n },\n {\n token: \"keyword\",\n regex: String.raw`(namespace|open|as|operation|function|body|adjoint|newtype|controlled|internal)\\b`,\n beginWord: true,\n },\n {\n token: \"keyword\",\n regex: String.raw`(if|elif|else|repeat|until|fixup|for|in|return|fail|within|apply)\\b`,\n beginWord: true,\n },\n {\n token: \"keyword\",\n regex: String.raw`(Adjoint|Controlled|Adj|Ctl|is|self|auto|distribute|invert|intrinsic)\\b`,\n beginWord: true,\n },\n {\n token: \"keyword\",\n regex: String.raw`(let|set|use|borrow|mutable)\\b`,\n beginWord: true,\n },\n {\n token: \"operatorKeyword\",\n regex: String.raw`(not|and|or)\\b|(w/)`,\n beginWord: true,\n },\n {\n token: \"operatorKeyword\",\n regex: String.raw`(=)|(!)|(<)|(>)|(\\+)|(-)|(\\*)|(/)|(\\^)|(%)|(\\|)|(&&&)|(~~~)|(\\.\\.\\.)|(\\.\\.)|(\\?)`,\n beginWord: false,\n },\n {\n token: \"meta\",\n regex: String.raw`(Int|BigInt|Double|Bool|Qubit|Pauli|Result|Range|String|Unit)\\b`,\n beginWord: true,\n },\n {\n token: \"atom\",\n regex: String.raw`(true|false|Pauli(I|X|Y|Z)|One|Zero)\\b`,\n beginWord: true,\n },\n ];\n let simpleRules = [];\n for (let rule of rules) {\n simpleRules.push({\n token: rule.token,\n regex: new RegExp(rule.regex, \"g\"),\n sol: rule.beginWord,\n });\n if (rule.beginWord) {\n // Need an additional rule due to the fact that CodeMirror simple mode doesn't work with ^ token\n simpleRules.push({\n token: rule.token,\n regex: new RegExp(String.raw`\\W` + rule.regex, \"g\"),\n sol: false,\n });\n }\n }\n\n // Register the mode defined above with CodeMirror\n window.CodeMirror.defineSimpleMode(\"qsharp\", { start: simpleRules });\n window.CodeMirror.defineMIME(\"text/x-qsharp\", \"qsharp\");\n\n // Tell Jupyter to associate %%qsharp magic cells with the qsharp mode\n window.Jupyter.CodeCell.options_default.highlight_modes[\"qsharp\"] = {\n reg: [/^%%qsharp/],\n };\n\n // Force re-highlighting of all cells the first time this code runs\n for (const cell of window.Jupyter.notebook.get_cells()) {\n cell.auto_highlight();\n }\n });\n}\n", |
| 30 | "text/plain": [] |
| 31 | }, |
| 32 | "metadata": {}, |
| 33 | "output_type": "display_data" |
| 34 | } |
| 35 | ], |
| 36 | "source": [ |
| 37 | "# Imports\n", |
| 38 | "from pathlib import Path\n", |
| 39 | "\n", |
| 40 | "import qdk\n", |
| 41 | "from qdk import qsharp\n", |
| 42 | "from qdk.qre import estimate, property_name, plot_estimates\n", |
| 43 | "from qdk.qre.models import GateBased, SurfaceCode, RoundBasedFactory\n", |
| 44 | "from qdk.qre.application import QSharpApplication\n", |
| 45 | "\n", |
| 46 | "from qdk.qre.instruction_ids import T, LATTICE_SURGERY\n", |
| 47 | "from qdk.qre.property_keys import NUM_TS_PER_ROTATION" |
| 48 | ] |
| 49 | }, |
| 50 | { |
| 51 | "cell_type": "markdown", |
| 52 | "id": "5f317aee", |
| 53 | "metadata": {}, |
| 54 | "source": [ |
| 55 | "## Setting up an application\n", |
| 56 | "\n", |
| 57 | "In this example, we use `QSharpApplication` to define the application we want to\n", |
| 58 | "estimate using a Q# entry point. Other application loaders are available to\n", |
| 59 | "[read quantum programs from other langauges and frameworks](1_qre_input.ipynb).\n", |
| 60 | "First, we load an existing Q# program from the samples directory and evaluate\n", |
| 61 | "it. Alternatively, we could load a Q# project or evaluate the Q# program inline\n", |
| 62 | "in a notebook cell. We create the application instance using the `Main()` entry\n", |
| 63 | "point of the Q# program." |
| 64 | ] |
| 65 | }, |
| 66 | { |
| 67 | "cell_type": "code", |
| 68 | "execution_count": 2, |
| 69 | "id": "5ecb972d", |
| 70 | "metadata": {}, |
| 71 | "outputs": [], |
| 72 | "source": [ |
| 73 | "# Evaluate the Q# application into the scope\n", |
| 74 | "qsharp.eval(Path(\"../algorithms/Ising/Simple1dIsingOrder1.qs\").read_text(encoding=\"utf-8\"))\n", |
| 75 | "\n", |
| 76 | "# The entry point in the Q# code above is called Main\n", |
| 77 | "app = QSharpApplication(qdk.code.Main)" |
| 78 | ] |
| 79 | }, |
| 80 | { |
| 81 | "cell_type": "markdown", |
| 82 | "id": "35416b3e", |
| 83 | "metadata": {}, |
| 84 | "source": [ |
| 85 | "We will soon use this application instance with the `estimate` method. This\n", |
| 86 | "will enumerate all possible traces given the application parameters and the\n", |
| 87 | "trace transform. But just to see some information of a trace, we can get the\n", |
| 88 | "first trace that is enumerated from the application and look at its required ISA\n", |
| 89 | "to implement it." |
| 90 | ] |
| 91 | }, |
| 92 | { |
| 93 | "cell_type": "code", |
| 94 | "execution_count": 3, |
| 95 | "id": "53268325", |
| 96 | "metadata": {}, |
| 97 | "outputs": [ |
| 98 | { |
| 99 | "data": { |
| 100 | "text/html": [ |
| 101 | "<div>\n", |
| 102 | "<style scoped>\n", |
| 103 | " .dataframe tbody tr th:only-of-type {\n", |
| 104 | " vertical-align: middle;\n", |
| 105 | " }\n", |
| 106 | "\n", |
| 107 | " .dataframe tbody tr th {\n", |
| 108 | " vertical-align: top;\n", |
| 109 | " }\n", |
| 110 | "\n", |
| 111 | " .dataframe thead th {\n", |
| 112 | " text-align: right;\n", |
| 113 | " }\n", |
| 114 | "</style>\n", |
| 115 | "<table border=\"1\" class=\"dataframe\">\n", |
| 116 | " <thead>\n", |
| 117 | " <tr style=\"text-align: right;\">\n", |
| 118 | " <th></th>\n", |
| 119 | " <th>encoding</th>\n", |
| 120 | " <th>arity</th>\n", |
| 121 | " </tr>\n", |
| 122 | " <tr>\n", |
| 123 | " <th>id</th>\n", |
| 124 | " <th></th>\n", |
| 125 | " <th></th>\n", |
| 126 | " </tr>\n", |
| 127 | " </thead>\n", |
| 128 | " <tbody>\n", |
| 129 | " <tr>\n", |
| 130 | " <th>RZ</th>\n", |
| 131 | " <td>LOGICAL</td>\n", |
| 132 | " <td>1</td>\n", |
| 133 | " </tr>\n", |
| 134 | " <tr>\n", |
| 135 | " <th>MEAS_Z</th>\n", |
| 136 | " <td>LOGICAL</td>\n", |
| 137 | " <td>1</td>\n", |
| 138 | " </tr>\n", |
| 139 | " </tbody>\n", |
| 140 | "</table>\n", |
| 141 | "</div>" |
| 142 | ], |
| 143 | "text/plain": [ |
| 144 | " encoding arity\n", |
| 145 | "id \n", |
| 146 | "RZ LOGICAL 1\n", |
| 147 | "MEAS_Z LOGICAL 1" |
| 148 | ] |
| 149 | }, |
| 150 | "execution_count": 3, |
| 151 | "metadata": {}, |
| 152 | "output_type": "execute_result" |
| 153 | } |
| 154 | ], |
| 155 | "source": [ |
| 156 | "# Get one of the generated traces (in fact, this application only generates one trace, but in general there could be multiple)\n", |
| 157 | "trace = next(app.enumerate_traces())\n", |
| 158 | "\n", |
| 159 | "trace.required_isa.as_frame()" |
| 160 | ] |
| 161 | }, |
| 162 | { |
| 163 | "cell_type": "markdown", |
| 164 | "id": "242a1b9b", |
| 165 | "metadata": {}, |
| 166 | "source": [ |
| 167 | "The output shows the logical instructions required by this trace. In this case, the algorithm needs `RZ` (arbitrary Z-rotation) and `MEAS_Z` (measurement in the Z basis), both operating on single qubits at the logical level. The resource estimator's job is to determine how many physical qubits and how much time are needed to implement these logical operations, given the target architecture and the chosen error correction scheme.\n", |
| 168 | "\n", |
| 169 | "Note that the `QSharpApplication` only computes logical counts required for Pauli-based computation (PSSPC)." |
| 170 | ] |
| 171 | }, |
| 172 | { |
| 173 | "cell_type": "markdown", |
| 174 | "id": "a2383f33", |
| 175 | "metadata": {}, |
| 176 | "source": [ |
| 177 | "## Choosing a target architecture\n", |
| 178 | "\n", |
| 179 | "The architecture model describes the physical quantum hardware. `GateBased` defines a gate-based architecture with configurable gate and measurement times (in nanoseconds) and a physical error rate (which defaults to $10^{-4}$). Calling `provided_isa()` on the architecture returns its physical instruction set: the set of native gates the hardware supports, along with each gate's encoding, arity, qubit cost, time, and error rate." |
| 180 | ] |
| 181 | }, |
| 182 | { |
| 183 | "cell_type": "code", |
| 184 | "execution_count": 4, |
| 185 | "id": "490eeeab", |
| 186 | "metadata": {}, |
| 187 | "outputs": [ |
| 188 | { |
| 189 | "data": { |
| 190 | "text/html": [ |
| 191 | "<div>\n", |
| 192 | "<style scoped>\n", |
| 193 | " .dataframe tbody tr th:only-of-type {\n", |
| 194 | " vertical-align: middle;\n", |
| 195 | " }\n", |
| 196 | "\n", |
| 197 | " .dataframe tbody tr th {\n", |
| 198 | " vertical-align: top;\n", |
| 199 | " }\n", |
| 200 | "\n", |
| 201 | " .dataframe thead th {\n", |
| 202 | " text-align: right;\n", |
| 203 | " }\n", |
| 204 | "</style>\n", |
| 205 | "<table border=\"1\" class=\"dataframe\">\n", |
| 206 | " <thead>\n", |
| 207 | " <tr style=\"text-align: right;\">\n", |
| 208 | " <th></th>\n", |
| 209 | " <th>encoding</th>\n", |
| 210 | " <th>arity</th>\n", |
| 211 | " <th>space</th>\n", |
| 212 | " <th>time</th>\n", |
| 213 | " <th>error</th>\n", |
| 214 | " </tr>\n", |
| 215 | " <tr>\n", |
| 216 | " <th>id</th>\n", |
| 217 | " <th></th>\n", |
| 218 | " <th></th>\n", |
| 219 | " <th></th>\n", |
| 220 | " <th></th>\n", |
| 221 | " <th></th>\n", |
| 222 | " </tr>\n", |
| 223 | " </thead>\n", |
| 224 | " <tbody>\n", |
| 225 | " <tr>\n", |
| 226 | " <th>PAULI_I</th>\n", |
| 227 | " <td>PHYSICAL</td>\n", |
| 228 | " <td>1</td>\n", |
| 229 | " <td>1</td>\n", |
| 230 | " <td>100</td>\n", |
| 231 | " <td>0.0001</td>\n", |
| 232 | " </tr>\n", |
| 233 | " <tr>\n", |
| 234 | " <th>SQRT_SQRT_X</th>\n", |
| 235 | " <td>PHYSICAL</td>\n", |
| 236 | " <td>1</td>\n", |
| 237 | " <td>1</td>\n", |
| 238 | " <td>100</td>\n", |
| 239 | " <td>0.0001</td>\n", |
| 240 | " </tr>\n", |
| 241 | " <tr>\n", |
| 242 | " <th>SQRT_SQRT_Y_DAG</th>\n", |
| 243 | " <td>PHYSICAL</td>\n", |
| 244 | " <td>1</td>\n", |
| 245 | " <td>1</td>\n", |
| 246 | " <td>100</td>\n", |
| 247 | " <td>0.0001</td>\n", |
| 248 | " </tr>\n", |
| 249 | " <tr>\n", |
| 250 | " <th>S</th>\n", |
| 251 | " <td>PHYSICAL</td>\n", |
| 252 | " <td>1</td>\n", |
| 253 | " <td>1</td>\n", |
| 254 | " <td>100</td>\n", |
| 255 | " <td>0.0001</td>\n", |
| 256 | " </tr>\n", |
| 257 | " <tr>\n", |
| 258 | " <th>SQRT_X_DAG</th>\n", |
| 259 | " <td>PHYSICAL</td>\n", |
| 260 | " <td>1</td>\n", |
| 261 | " <td>1</td>\n", |
| 262 | " <td>100</td>\n", |
| 263 | " <td>0.0001</td>\n", |
| 264 | " </tr>\n", |
| 265 | " <tr>\n", |
| 266 | " <th>MEAS_X</th>\n", |
| 267 | " <td>PHYSICAL</td>\n", |
| 268 | " <td>1</td>\n", |
| 269 | " <td>1</td>\n", |
| 270 | " <td>500</td>\n", |
| 271 | " <td>0.0001</td>\n", |
| 272 | " </tr>\n", |
| 273 | " <tr>\n", |
| 274 | " <th>RZ</th>\n", |
| 275 | " <td>PHYSICAL</td>\n", |
| 276 | " <td>1</td>\n", |
| 277 | " <td>1</td>\n", |
| 278 | " <td>100</td>\n", |
| 279 | " <td>0.0001</td>\n", |
| 280 | " </tr>\n", |
| 281 | " <tr>\n", |
| 282 | " <th>T_DAG</th>\n", |
| 283 | " <td>PHYSICAL</td>\n", |
| 284 | " <td>1</td>\n", |
| 285 | " <td>1</td>\n", |
| 286 | " <td>100</td>\n", |
| 287 | " <td>0.0001</td>\n", |
| 288 | " </tr>\n", |
| 289 | " <tr>\n", |
| 290 | " <th>PAULI_Y</th>\n", |
| 291 | " <td>PHYSICAL</td>\n", |
| 292 | " <td>1</td>\n", |
| 293 | " <td>1</td>\n", |
| 294 | " <td>100</td>\n", |
| 295 | " <td>0.0001</td>\n", |
| 296 | " </tr>\n", |
| 297 | " <tr>\n", |
| 298 | " <th>SQRT_SQRT_Y</th>\n", |
| 299 | " <td>PHYSICAL</td>\n", |
| 300 | " <td>1</td>\n", |
| 301 | " <td>1</td>\n", |
| 302 | " <td>100</td>\n", |
| 303 | " <td>0.0001</td>\n", |
| 304 | " </tr>\n", |
| 305 | " <tr>\n", |
| 306 | " <th>MEAS_Z</th>\n", |
| 307 | " <td>PHYSICAL</td>\n", |
| 308 | " <td>1</td>\n", |
| 309 | " <td>1</td>\n", |
| 310 | " <td>500</td>\n", |
| 311 | " <td>0.0001</td>\n", |
| 312 | " </tr>\n", |
| 313 | " <tr>\n", |
| 314 | " <th>CNOT</th>\n", |
| 315 | " <td>PHYSICAL</td>\n", |
| 316 | " <td>2</td>\n", |
| 317 | " <td>2</td>\n", |
| 318 | " <td>100</td>\n", |
| 319 | " <td>0.0001</td>\n", |
| 320 | " </tr>\n", |
| 321 | " <tr>\n", |
| 322 | " <th>SQRT_Y_DAG</th>\n", |
| 323 | " <td>PHYSICAL</td>\n", |
| 324 | " <td>1</td>\n", |
| 325 | " <td>1</td>\n", |
| 326 | " <td>100</td>\n", |
| 327 | " <td>0.0001</td>\n", |
| 328 | " </tr>\n", |
| 329 | " <tr>\n", |
| 330 | " <th>SQRT_X</th>\n", |
| 331 | " <td>PHYSICAL</td>\n", |
| 332 | " <td>1</td>\n", |
| 333 | " <td>1</td>\n", |
| 334 | " <td>100</td>\n", |
| 335 | " <td>0.0001</td>\n", |
| 336 | " </tr>\n", |
| 337 | " <tr>\n", |
| 338 | " <th>H</th>\n", |
| 339 | " <td>PHYSICAL</td>\n", |
| 340 | " <td>1</td>\n", |
| 341 | " <td>1</td>\n", |
| 342 | " <td>100</td>\n", |
| 343 | " <td>0.0001</td>\n", |
| 344 | " </tr>\n", |
| 345 | " <tr>\n", |
| 346 | " <th>RY</th>\n", |
| 347 | " <td>PHYSICAL</td>\n", |
| 348 | " <td>1</td>\n", |
| 349 | " <td>1</td>\n", |
| 350 | " <td>100</td>\n", |
| 351 | " <td>0.0001</td>\n", |
| 352 | " </tr>\n", |
| 353 | " <tr>\n", |
| 354 | " <th>T</th>\n", |
| 355 | " <td>PHYSICAL</td>\n", |
| 356 | " <td>1</td>\n", |
| 357 | " <td>1</td>\n", |
| 358 | " <td>100</td>\n", |
| 359 | " <td>0.0001</td>\n", |
| 360 | " </tr>\n", |
| 361 | " <tr>\n", |
| 362 | " <th>PAULI_X</th>\n", |
| 363 | " <td>PHYSICAL</td>\n", |
| 364 | " <td>1</td>\n", |
| 365 | " <td>1</td>\n", |
| 366 | " <td>100</td>\n", |
| 367 | " <td>0.0001</td>\n", |
| 368 | " </tr>\n", |
| 369 | " <tr>\n", |
| 370 | " <th>SQRT_SQRT_X_DAG</th>\n", |
| 371 | " <td>PHYSICAL</td>\n", |
| 372 | " <td>1</td>\n", |
| 373 | " <td>1</td>\n", |
| 374 | " <td>100</td>\n", |
| 375 | " <td>0.0001</td>\n", |
| 376 | " </tr>\n", |
| 377 | " <tr>\n", |
| 378 | " <th>MEAS_Y</th>\n", |
| 379 | " <td>PHYSICAL</td>\n", |
| 380 | " <td>1</td>\n", |
| 381 | " <td>1</td>\n", |
| 382 | " <td>500</td>\n", |
| 383 | " <td>0.0001</td>\n", |
| 384 | " </tr>\n", |
| 385 | " <tr>\n", |
| 386 | " <th>S_DAG</th>\n", |
| 387 | " <td>PHYSICAL</td>\n", |
| 388 | " <td>1</td>\n", |
| 389 | " <td>1</td>\n", |
| 390 | " <td>100</td>\n", |
| 391 | " <td>0.0001</td>\n", |
| 392 | " </tr>\n", |
| 393 | " <tr>\n", |
| 394 | " <th>SQRT_Y</th>\n", |
| 395 | " <td>PHYSICAL</td>\n", |
| 396 | " <td>1</td>\n", |
| 397 | " <td>1</td>\n", |
| 398 | " <td>100</td>\n", |
| 399 | " <td>0.0001</td>\n", |
| 400 | " </tr>\n", |
| 401 | " <tr>\n", |
| 402 | " <th>CZ</th>\n", |
| 403 | " <td>PHYSICAL</td>\n", |
| 404 | " <td>2</td>\n", |
| 405 | " <td>2</td>\n", |
| 406 | " <td>100</td>\n", |
| 407 | " <td>0.0001</td>\n", |
| 408 | " </tr>\n", |
| 409 | " <tr>\n", |
| 410 | " <th>RX</th>\n", |
| 411 | " <td>PHYSICAL</td>\n", |
| 412 | " <td>1</td>\n", |
| 413 | " <td>1</td>\n", |
| 414 | " <td>100</td>\n", |
| 415 | " <td>0.0001</td>\n", |
| 416 | " </tr>\n", |
| 417 | " <tr>\n", |
| 418 | " <th>PAULI_Z</th>\n", |
| 419 | " <td>PHYSICAL</td>\n", |
| 420 | " <td>1</td>\n", |
| 421 | " <td>1</td>\n", |
| 422 | " <td>100</td>\n", |
| 423 | " <td>0.0001</td>\n", |
| 424 | " </tr>\n", |
| 425 | " </tbody>\n", |
| 426 | "</table>\n", |
| 427 | "</div>" |
| 428 | ], |
| 429 | "text/plain": [ |
| 430 | " encoding arity space time error\n", |
| 431 | "id \n", |
| 432 | "PAULI_I PHYSICAL 1 1 100 0.0001\n", |
| 433 | "SQRT_SQRT_X PHYSICAL 1 1 100 0.0001\n", |
| 434 | "SQRT_SQRT_Y_DAG PHYSICAL 1 1 100 0.0001\n", |
| 435 | "S PHYSICAL 1 1 100 0.0001\n", |
| 436 | "SQRT_X_DAG PHYSICAL 1 1 100 0.0001\n", |
| 437 | "MEAS_X PHYSICAL 1 1 500 0.0001\n", |
| 438 | "RZ PHYSICAL 1 1 100 0.0001\n", |
| 439 | "T_DAG PHYSICAL 1 1 100 0.0001\n", |
| 440 | "PAULI_Y PHYSICAL 1 1 100 0.0001\n", |
| 441 | "SQRT_SQRT_Y PHYSICAL 1 1 100 0.0001\n", |
| 442 | "MEAS_Z PHYSICAL 1 1 500 0.0001\n", |
| 443 | "CNOT PHYSICAL 2 2 100 0.0001\n", |
| 444 | "SQRT_Y_DAG PHYSICAL 1 1 100 0.0001\n", |
| 445 | "SQRT_X PHYSICAL 1 1 100 0.0001\n", |
| 446 | "H PHYSICAL 1 1 100 0.0001\n", |
| 447 | "RY PHYSICAL 1 1 100 0.0001\n", |
| 448 | "T PHYSICAL 1 1 100 0.0001\n", |
| 449 | "PAULI_X PHYSICAL 1 1 100 0.0001\n", |
| 450 | "SQRT_SQRT_X_DAG PHYSICAL 1 1 100 0.0001\n", |
| 451 | "MEAS_Y PHYSICAL 1 1 500 0.0001\n", |
| 452 | "S_DAG PHYSICAL 1 1 100 0.0001\n", |
| 453 | "SQRT_Y PHYSICAL 1 1 100 0.0001\n", |
| 454 | "CZ PHYSICAL 2 2 100 0.0001\n", |
| 455 | "RX PHYSICAL 1 1 100 0.0001\n", |
| 456 | "PAULI_Z PHYSICAL 1 1 100 0.0001" |
| 457 | ] |
| 458 | }, |
| 459 | "execution_count": 4, |
| 460 | "metadata": {}, |
| 461 | "output_type": "execute_result" |
| 462 | } |
| 463 | ], |
| 464 | "source": [ |
| 465 | "# Define a gate-based architecture: physical error rate 1e-4, gate time 100 ns, measurement 500 ns\n", |
| 466 | "arch = GateBased(error_rate=1e-4, gate_time=100, measurement_time=500)\n", |
| 467 | "arch.provided_isa(arch.context()).as_frame()" |
| 468 | ] |
| 469 | }, |
| 470 | { |
| 471 | "cell_type": "markdown", |
| 472 | "id": "436bd2b5", |
| 473 | "metadata": {}, |
| 474 | "source": [ |
| 475 | "The table above shows the physical gates provided by this architecture. All gates operate at the `PHYSICAL` encoding level. Single-qubit gates (e.g., `H`, `T`) each use 1 physical qubit and take 100 ns, while two-qubit gates (e.g., `CNOT`, `CZ`) use 2 qubits and also take 100 ns. Measurement (`MEAS_Z`) is slower at 500 ns. All gates share the same physical error rate of $10^{-4}$." |
| 476 | ] |
| 477 | }, |
| 478 | { |
| 479 | "cell_type": "markdown", |
| 480 | "id": "476a9aa1", |
| 481 | "metadata": {}, |
| 482 | "source": [ |
| 483 | "## Estimate\n", |
| 484 | "\n", |
| 485 | "The `estimate` function takes the application and architecture as inputs, along with an ISA query that specifies which quantum error correction codes and magic state factories to explore. The `SurfaceCode.q() * RoundBasedFactory.q()` ISA query enumerates the Cartesian product of surface code configurations (varying code distances) and round-based magic state distillation factories (varying distillation protocols). The `max_error` parameter sets the maximum tolerable error probability for the overall computation.\n", |
| 486 | "\n", |
| 487 | "The function returns an `EstimationTable` containing only Pareto-optimal results. These are configurations where no other result is simultaneously better in both physical qubits and runtime while staying within the error budget. The default columns show the total physical qubit count, the total runtime, and the accumulated error for each result." |
| 488 | ] |
| 489 | }, |
| 490 | { |
| 491 | "cell_type": "code", |
| 492 | "execution_count": null, |
| 493 | "id": "ffa0d10b", |
| 494 | "metadata": {}, |
| 495 | "outputs": [ |
| 496 | { |
| 497 | "data": { |
| 498 | "text/html": [ |
| 499 | "<div>\n", |
| 500 | "<style scoped>\n", |
| 501 | " .dataframe tbody tr th:only-of-type {\n", |
| 502 | " vertical-align: middle;\n", |
| 503 | " }\n", |
| 504 | "\n", |
| 505 | " .dataframe tbody tr th {\n", |
| 506 | " vertical-align: top;\n", |
| 507 | " }\n", |
| 508 | "\n", |
| 509 | " .dataframe thead th {\n", |
| 510 | " text-align: right;\n", |
| 511 | " }\n", |
| 512 | "</style>\n", |
| 513 | "<table border=\"1\" class=\"dataframe\">\n", |
| 514 | " <thead>\n", |
| 515 | " <tr style=\"text-align: right;\">\n", |
| 516 | " <th></th>\n", |
| 517 | " <th>qubits</th>\n", |
| 518 | " <th>runtime</th>\n", |
| 519 | " <th>error</th>\n", |
| 520 | " </tr>\n", |
| 521 | " </thead>\n", |
| 522 | " <tbody>\n", |
| 523 | " <tr>\n", |
| 524 | " <th>0</th>\n", |
| 525 | " <td>29036</td>\n", |
| 526 | " <td>0 days 00:00:00.005213</td>\n", |
| 527 | " <td>0.003162</td>\n", |
| 528 | " </tr>\n", |
| 529 | " <tr>\n", |
| 530 | " <th>1</th>\n", |
| 531 | " <td>30268</td>\n", |
| 532 | " <td>0 days 00:00:00.004411</td>\n", |
| 533 | " <td>0.003162</td>\n", |
| 534 | " </tr>\n", |
| 535 | " <tr>\n", |
| 536 | " <th>2</th>\n", |
| 537 | " <td>32928</td>\n", |
| 538 | " <td>0 days 00:00:00.003609</td>\n", |
| 539 | " <td>0.003162</td>\n", |
| 540 | " </tr>\n", |
| 541 | " <tr>\n", |
| 542 | " <th>3</th>\n", |
| 543 | " <td>37996</td>\n", |
| 544 | " <td>0 days 00:00:00.002807</td>\n", |
| 545 | " <td>0.003166</td>\n", |
| 546 | " </tr>\n", |
| 547 | " <tr>\n", |
| 548 | " <th>4</th>\n", |
| 549 | " <td>52332</td>\n", |
| 550 | " <td>0 days 00:00:00.002005</td>\n", |
| 551 | " <td>0.003499</td>\n", |
| 552 | " </tr>\n", |
| 553 | " </tbody>\n", |
| 554 | "</table>\n", |
| 555 | "</div>" |
| 556 | ], |
| 557 | "text/plain": [ |
| 558 | " qubits runtime error\n", |
| 559 | "0 29036 0 days 00:00:00.005213 0.003162\n", |
| 560 | "1 30268 0 days 00:00:00.004411 0.003162\n", |
| 561 | "2 32928 0 days 00:00:00.003609 0.003162\n", |
| 562 | "3 37996 0 days 00:00:00.002807 0.003166\n", |
| 563 | "4 52332 0 days 00:00:00.002005 0.003499" |
| 564 | ] |
| 565 | }, |
| 566 | "execution_count": 5, |
| 567 | "metadata": {}, |
| 568 | "output_type": "execute_result" |
| 569 | } |
| 570 | ], |
| 571 | "source": [ |
| 572 | "# Run estimation over surface code × round-based factory configurations with 1% error budget\n", |
| 573 | "results = estimate(app, arch, isa_query=SurfaceCode.q() * RoundBasedFactory.q(), max_error=0.01)\n", |
| 574 | "\n", |
| 575 | "# Returns a pandas DataFrame that can be used for further analysis or visualization\n", |
| 576 | "results.as_frame()" |
| 577 | ] |
| 578 | }, |
| 579 | { |
| 580 | "cell_type": "markdown", |
| 581 | "id": "d41b7893", |
| 582 | "metadata": {}, |
| 583 | "source": [ |
| 584 | "Each row in the table represents a Pareto-optimal configuration. The first row uses the fewest physical qubits but has the longest runtime, while the last row is the fastest but requires the most qubits. All results stay within the specified `max_error=0.01` error budget.\n", |
| 585 | "\n", |
| 586 | "We can inspect the estimation statistics to see how large the design space was and how many configurations were explored to produce these Pareto-optimal results." |
| 587 | ] |
| 588 | }, |
| 589 | { |
| 590 | "cell_type": "code", |
| 591 | "execution_count": 6, |
| 592 | "id": "5354ea4a", |
| 593 | "metadata": {}, |
| 594 | "outputs": [ |
| 595 | { |
| 596 | "data": { |
| 597 | "text/plain": [ |
| 598 | "EstimationTableStats(num_traces=32, num_isas=1248, total_jobs=17424, successful_estimates=353, pareto_results=5)" |
| 599 | ] |
| 600 | }, |
| 601 | "execution_count": 6, |
| 602 | "metadata": {}, |
| 603 | "output_type": "execute_result" |
| 604 | } |
| 605 | ], |
| 606 | "source": [ |
| 607 | "results.stats" |
| 608 | ] |
| 609 | }, |
| 610 | { |
| 611 | "cell_type": "markdown", |
| 612 | "id": "06a3c26b", |
| 613 | "metadata": {}, |
| 614 | "source": [ |
| 615 | "### Adding pre-configured columns to the table\n", |
| 616 | "\n", |
| 617 | "The `EstimationTable` provides convenience methods to add commonly used columns. The method `add_qubit_partition_column()` breaks down the total physical qubit count into three components: compute qubits (for logical operations), factory qubits (for magic state distillation), and memory qubits (for data storage). The method `add_factory_summary_column()` adds a human-readable summary of the magic state factories used in each result (e.g., `20×T` meaning 20 copies of a T-state factory). These numbers can help to understand whether space-time trade-offs are possible. For example, if the amount of factory qubits is relatively high compared to the total number of qubits and many factory copies are used, than it's possible to reduce the total number of qubits by letting the algorithm run longer." |
| 618 | ] |
| 619 | }, |
| 620 | { |
| 621 | "cell_type": "code", |
| 622 | "execution_count": null, |
| 623 | "id": "621904f7", |
| 624 | "metadata": {}, |
| 625 | "outputs": [ |
| 626 | { |
| 627 | "data": { |
| 628 | "text/html": [ |
| 629 | "<div>\n", |
| 630 | "<style scoped>\n", |
| 631 | " .dataframe tbody tr th:only-of-type {\n", |
| 632 | " vertical-align: middle;\n", |
| 633 | " }\n", |
| 634 | "\n", |
| 635 | " .dataframe tbody tr th {\n", |
| 636 | " vertical-align: top;\n", |
| 637 | " }\n", |
| 638 | "\n", |
| 639 | " .dataframe thead th {\n", |
| 640 | " text-align: right;\n", |
| 641 | " }\n", |
| 642 | "</style>\n", |
| 643 | "<table border=\"1\" class=\"dataframe\">\n", |
| 644 | " <thead>\n", |
| 645 | " <tr style=\"text-align: right;\">\n", |
| 646 | " <th></th>\n", |
| 647 | " <th>qubits</th>\n", |
| 648 | " <th>runtime</th>\n", |
| 649 | " <th>error</th>\n", |
| 650 | " <th>physical_compute_qubits</th>\n", |
| 651 | " <th>physical_factory_qubits</th>\n", |
| 652 | " <th>physical_memory_qubits</th>\n", |
| 653 | " <th>factories</th>\n", |
| 654 | " </tr>\n", |
| 655 | " </thead>\n", |
| 656 | " <tbody>\n", |
| 657 | " <tr>\n", |
| 658 | " <th>0</th>\n", |
| 659 | " <td>29036</td>\n", |
| 660 | " <td>0 days 00:00:00.005213</td>\n", |
| 661 | " <td>0.003162</td>\n", |
| 662 | " <td>9436</td>\n", |
| 663 | " <td>19600</td>\n", |
| 664 | " <td>0</td>\n", |
| 665 | " <td>20×T</td>\n", |
| 666 | " </tr>\n", |
| 667 | " <tr>\n", |
| 668 | " <th>1</th>\n", |
| 669 | " <td>30268</td>\n", |
| 670 | " <td>0 days 00:00:00.004411</td>\n", |
| 671 | " <td>0.003162</td>\n", |
| 672 | " <td>6748</td>\n", |
| 673 | " <td>23520</td>\n", |
| 674 | " <td>0</td>\n", |
| 675 | " <td>24×T</td>\n", |
| 676 | " </tr>\n", |
| 677 | " <tr>\n", |
| 678 | " <th>2</th>\n", |
| 679 | " <td>32928</td>\n", |
| 680 | " <td>0 days 00:00:00.003609</td>\n", |
| 681 | " <td>0.003162</td>\n", |
| 682 | " <td>4508</td>\n", |
| 683 | " <td>28420</td>\n", |
| 684 | " <td>0</td>\n", |
| 685 | " <td>29×T</td>\n", |
| 686 | " </tr>\n", |
| 687 | " <tr>\n", |
| 688 | " <th>3</th>\n", |
| 689 | " <td>37996</td>\n", |
| 690 | " <td>0 days 00:00:00.002807</td>\n", |
| 691 | " <td>0.003166</td>\n", |
| 692 | " <td>2716</td>\n", |
| 693 | " <td>35280</td>\n", |
| 694 | " <td>0</td>\n", |
| 695 | " <td>36×T</td>\n", |
| 696 | " </tr>\n", |
| 697 | " <tr>\n", |
| 698 | " <th>4</th>\n", |
| 699 | " <td>52332</td>\n", |
| 700 | " <td>0 days 00:00:00.002005</td>\n", |
| 701 | " <td>0.003499</td>\n", |
| 702 | " <td>1372</td>\n", |
| 703 | " <td>50960</td>\n", |
| 704 | " <td>0</td>\n", |
| 705 | " <td>52×T</td>\n", |
| 706 | " </tr>\n", |
| 707 | " </tbody>\n", |
| 708 | "</table>\n", |
| 709 | "</div>" |
| 710 | ], |
| 711 | "text/plain": [ |
| 712 | " qubits runtime error physical_compute_qubits \\\n", |
| 713 | "0 29036 0 days 00:00:00.005213 0.003162 9436 \n", |
| 714 | "1 30268 0 days 00:00:00.004411 0.003162 6748 \n", |
| 715 | "2 32928 0 days 00:00:00.003609 0.003162 4508 \n", |
| 716 | "3 37996 0 days 00:00:00.002807 0.003166 2716 \n", |
| 717 | "4 52332 0 days 00:00:00.002005 0.003499 1372 \n", |
| 718 | "\n", |
| 719 | " physical_factory_qubits physical_memory_qubits factories \n", |
| 720 | "0 19600 0 20×T \n", |
| 721 | "1 23520 0 24×T \n", |
| 722 | "2 28420 0 29×T \n", |
| 723 | "3 35280 0 36×T \n", |
| 724 | "4 50960 0 52×T " |
| 725 | ] |
| 726 | }, |
| 727 | "execution_count": 7, |
| 728 | "metadata": {}, |
| 729 | "output_type": "execute_result" |
| 730 | } |
| 731 | ], |
| 732 | "source": [ |
| 733 | "# Break total qubits into compute, factory, and memory components\n", |
| 734 | "results.add_qubit_partition_column()\n", |
| 735 | "# Add a human-readable factory summary (e.g., \"20×T\")\n", |
| 736 | "results.add_factory_summary_column()\n", |
| 737 | "\n", |
| 738 | "results.as_frame()" |
| 739 | ] |
| 740 | }, |
| 741 | { |
| 742 | "cell_type": "markdown", |
| 743 | "id": "48cac642", |
| 744 | "metadata": {}, |
| 745 | "source": [ |
| 746 | "### Querying properties of selected results\n", |
| 747 | "\n", |
| 748 | "Individual results can be accessed by index (e.g., `results[0]` for the first Pareto-optimal result). Each result exposes:\n", |
| 749 | "- `properties`: a dictionary of named metrics such as qubit counts and algorithm parameters,\n", |
| 750 | "- `source`: a mapping from instruction IDs to their implementation details, including error rates and timing, and\n", |
| 751 | "- `factories`: a mapping from instruction IDs to their magic state factory details, including the number of states produced." |
| 752 | ] |
| 753 | }, |
| 754 | { |
| 755 | "cell_type": "code", |
| 756 | "execution_count": 8, |
| 757 | "id": "e1df03eb", |
| 758 | "metadata": {}, |
| 759 | "outputs": [ |
| 760 | { |
| 761 | "name": "stdout", |
| 762 | "output_type": "stream", |
| 763 | "text": [ |
| 764 | "{'LOGICAL_MEMORY_QUBITS': 0, 'EVALUATION_TIME': 2032086, 'NUM_TS_PER_ROTATION': 13, 'ALGORITHM_COMPUTE_QUBITS': 9, 'PHYSICAL_COMPUTE_QUBITS': 9436, 'ALGORITHM_MEMORY_QUBITS': 0, 'LOGICAL_COMPUTE_QUBITS': 28, 'PHYSICAL_FACTORY_QUBITS': 19600}\n", |
| 765 | "2.1303500000000005e-07\n", |
| 766 | "1547\n" |
| 767 | ] |
| 768 | } |
| 769 | ], |
| 770 | "source": [ |
| 771 | "# Print all properties for the first result, converting numeric keys to human-readable names\n", |
| 772 | "print(dict((property_name(k), v) for k, v in results[0].properties.items()))\n", |
| 773 | "\n", |
| 774 | "# Get the logical error rate of the T gate instruction in the first result\n", |
| 775 | "print(results[0].source[T].instruction.error_rate())\n", |
| 776 | "\n", |
| 777 | "# Get the total number of T states produced by the magic state factory\n", |
| 778 | "print(results[0].factories[T].states)" |
| 779 | ] |
| 780 | }, |
| 781 | { |
| 782 | "cell_type": "markdown", |
| 783 | "id": "bf9c7111", |
| 784 | "metadata": {}, |
| 785 | "source": [ |
| 786 | "### Adding custom columns to the table\n", |
| 787 | "\n", |
| 788 | "Beyond the pre-configured columns, you can add arbitrary custom columns using `add_column(name, function)` or populate a column from a property key using `add_property_column(key)`. With `add_column`, the function receives an `EstimationTableEntry` and returns the value for that column. With `add_property_column`, the column is automatically populated from the entry's `properties` dictionary using the given key. These methods are useful for surfacing specific metrics, such as per-instruction error rates, factory state counts, or algorithm-level parameters, directly in the results table." |
| 789 | ] |
| 790 | }, |
| 791 | { |
| 792 | "cell_type": "code", |
| 793 | "execution_count": 9, |
| 794 | "id": "e1fcc7d3", |
| 795 | "metadata": {}, |
| 796 | "outputs": [ |
| 797 | { |
| 798 | "data": { |
| 799 | "text/html": [ |
| 800 | "<div>\n", |
| 801 | "<style scoped>\n", |
| 802 | " .dataframe tbody tr th:only-of-type {\n", |
| 803 | " vertical-align: middle;\n", |
| 804 | " }\n", |
| 805 | "\n", |
| 806 | " .dataframe tbody tr th {\n", |
| 807 | " vertical-align: top;\n", |
| 808 | " }\n", |
| 809 | "\n", |
| 810 | " .dataframe thead th {\n", |
| 811 | " text-align: right;\n", |
| 812 | " }\n", |
| 813 | "</style>\n", |
| 814 | "<table border=\"1\" class=\"dataframe\">\n", |
| 815 | " <thead>\n", |
| 816 | " <tr style=\"text-align: right;\">\n", |
| 817 | " <th></th>\n", |
| 818 | " <th>qubits</th>\n", |
| 819 | " <th>runtime</th>\n", |
| 820 | " <th>error</th>\n", |
| 821 | " <th>physical_compute_qubits</th>\n", |
| 822 | " <th>physical_factory_qubits</th>\n", |
| 823 | " <th>physical_memory_qubits</th>\n", |
| 824 | " <th>factories</th>\n", |
| 825 | " <th>t_error_rate</th>\n", |
| 826 | " <th>logical_error_rate</th>\n", |
| 827 | " <th>t_states</th>\n", |
| 828 | " <th>num_ts_per_rotation</th>\n", |
| 829 | " </tr>\n", |
| 830 | " </thead>\n", |
| 831 | " <tbody>\n", |
| 832 | " <tr>\n", |
| 833 | " <th>0</th>\n", |
| 834 | " <td>29036</td>\n", |
| 835 | " <td>0 days 00:00:00.005213</td>\n", |
| 836 | " <td>0.003162</td>\n", |
| 837 | " <td>9436</td>\n", |
| 838 | " <td>19600</td>\n", |
| 839 | " <td>0</td>\n", |
| 840 | " <td>20×T</td>\n", |
| 841 | " <td>2.130350e-07</td>\n", |
| 842 | " <td>3.000000e-16</td>\n", |
| 843 | " <td>1547</td>\n", |
| 844 | " <td>13</td>\n", |
| 845 | " </tr>\n", |
| 846 | " <tr>\n", |
| 847 | " <th>1</th>\n", |
| 848 | " <td>30268</td>\n", |
| 849 | " <td>0 days 00:00:00.004411</td>\n", |
| 850 | " <td>0.003162</td>\n", |
| 851 | " <td>6748</td>\n", |
| 852 | " <td>23520</td>\n", |
| 853 | " <td>0</td>\n", |
| 854 | " <td>24×T</td>\n", |
| 855 | " <td>2.130350e-07</td>\n", |
| 856 | " <td>3.000000e-14</td>\n", |
| 857 | " <td>1547</td>\n", |
| 858 | " <td>13</td>\n", |
| 859 | " </tr>\n", |
| 860 | " <tr>\n", |
| 861 | " <th>2</th>\n", |
| 862 | " <td>32928</td>\n", |
| 863 | " <td>0 days 00:00:00.003609</td>\n", |
| 864 | " <td>0.003162</td>\n", |
| 865 | " <td>4508</td>\n", |
| 866 | " <td>28420</td>\n", |
| 867 | " <td>0</td>\n", |
| 868 | " <td>29×T</td>\n", |
| 869 | " <td>2.130350e-07</td>\n", |
| 870 | " <td>3.000000e-12</td>\n", |
| 871 | " <td>1547</td>\n", |
| 872 | " <td>13</td>\n", |
| 873 | " </tr>\n", |
| 874 | " <tr>\n", |
| 875 | " <th>3</th>\n", |
| 876 | " <td>37996</td>\n", |
| 877 | " <td>0 days 00:00:00.002807</td>\n", |
| 878 | " <td>0.003166</td>\n", |
| 879 | " <td>2716</td>\n", |
| 880 | " <td>35280</td>\n", |
| 881 | " <td>0</td>\n", |
| 882 | " <td>36×T</td>\n", |
| 883 | " <td>2.130350e-07</td>\n", |
| 884 | " <td>3.000000e-10</td>\n", |
| 885 | " <td>1547</td>\n", |
| 886 | " <td>13</td>\n", |
| 887 | " </tr>\n", |
| 888 | " <tr>\n", |
| 889 | " <th>4</th>\n", |
| 890 | " <td>52332</td>\n", |
| 891 | " <td>0 days 00:00:00.002005</td>\n", |
| 892 | " <td>0.003499</td>\n", |
| 893 | " <td>1372</td>\n", |
| 894 | " <td>50960</td>\n", |
| 895 | " <td>0</td>\n", |
| 896 | " <td>52×T</td>\n", |
| 897 | " <td>2.130350e-07</td>\n", |
| 898 | " <td>3.000000e-08</td>\n", |
| 899 | " <td>1547</td>\n", |
| 900 | " <td>13</td>\n", |
| 901 | " </tr>\n", |
| 902 | " </tbody>\n", |
| 903 | "</table>\n", |
| 904 | "</div>" |
| 905 | ], |
| 906 | "text/plain": [ |
| 907 | " qubits runtime error physical_compute_qubits \\\n", |
| 908 | "0 29036 0 days 00:00:00.005213 0.003162 9436 \n", |
| 909 | "1 30268 0 days 00:00:00.004411 0.003162 6748 \n", |
| 910 | "2 32928 0 days 00:00:00.003609 0.003162 4508 \n", |
| 911 | "3 37996 0 days 00:00:00.002807 0.003166 2716 \n", |
| 912 | "4 52332 0 days 00:00:00.002005 0.003499 1372 \n", |
| 913 | "\n", |
| 914 | " physical_factory_qubits physical_memory_qubits factories t_error_rate \\\n", |
| 915 | "0 19600 0 20×T 2.130350e-07 \n", |
| 916 | "1 23520 0 24×T 2.130350e-07 \n", |
| 917 | "2 28420 0 29×T 2.130350e-07 \n", |
| 918 | "3 35280 0 36×T 2.130350e-07 \n", |
| 919 | "4 50960 0 52×T 2.130350e-07 \n", |
| 920 | "\n", |
| 921 | " logical_error_rate t_states num_ts_per_rotation \n", |
| 922 | "0 3.000000e-16 1547 13 \n", |
| 923 | "1 3.000000e-14 1547 13 \n", |
| 924 | "2 3.000000e-12 1547 13 \n", |
| 925 | "3 3.000000e-10 1547 13 \n", |
| 926 | "4 3.000000e-08 1547 13 " |
| 927 | ] |
| 928 | }, |
| 929 | "execution_count": 9, |
| 930 | "metadata": {}, |
| 931 | "output_type": "execute_result" |
| 932 | } |
| 933 | ], |
| 934 | "source": [ |
| 935 | "# Error rate of the T instruction (fixed arity, so no argument needed)\n", |
| 936 | "results.add_column(\"t_error_rate\", lambda r: r.source[T].instruction.error_rate())\n", |
| 937 | "\n", |
| 938 | "# Error rate of the LATTICE_SURGERY instruction. This instruction has variable\n", |
| 939 | "# arity (it operates on a varying number of qubits), so we must pass an arity\n", |
| 940 | "# value to evaluate the error rate function. Arity 1 is the canonical single-\n", |
| 941 | "# qubit evaluation point used throughout the estimator for comparison.\n", |
| 942 | "results.add_column(\"logical_error_rate\", lambda r: r.source[LATTICE_SURGERY].instruction.error_rate(1))\n", |
| 943 | "\n", |
| 944 | "# Number of T states produced by the magic state factory\n", |
| 945 | "results.add_column(\"t_states\", lambda r: r.factories[T].states)\n", |
| 946 | "\n", |
| 947 | "# Add a column directly from a property key\n", |
| 948 | "results.add_property_column(NUM_TS_PER_ROTATION)\n", |
| 949 | "\n", |
| 950 | "results.as_frame()" |
| 951 | ] |
| 952 | }, |
| 953 | { |
| 954 | "cell_type": "markdown", |
| 955 | "id": "6a7569b8", |
| 956 | "metadata": {}, |
| 957 | "source": [ |
| 958 | "### Plotting the results\n", |
| 959 | "\n", |
| 960 | "The `plot` method produces a log-log scatter plot of the Pareto frontier, with runtime on the x-axis and physical qubits on the y-axis. The `runtime_unit` parameter controls the x-axis scale (e.g., `\"ms\"`, `\"s\"`, `\"hours\"`), and `figsize` sets the figure dimensions." |
| 961 | ] |
| 962 | }, |
| 963 | { |
| 964 | "cell_type": "code", |
| 965 | "execution_count": 10, |
| 966 | "id": "dd244938", |
| 967 | "metadata": {}, |
| 968 | "outputs": [ |
| 969 | { |
| 970 | "data": { |
| 971 | "image/png": "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iTDjFunp6Ro0aJDat2+va665RsnJydq/f79Lx9i4caOSkpIUFhYmi8Wi999/v8F+GRkZ6tmzp/z9/TVkyBBt27bNtq2goEDdu3e3ve7evbv+8Y9/uLROmNvChQsVExOjoKAgBQUFadiwYfrkk09cOgbf5dZv3rx5slgseuKJJ1x6XD57uAphxg02bNig1NRUbd26VWvWrFFNTY1++ctfqqKiosH+mzdvVk1NjUP73r17VVRU1OA+FRUVslqtysjIaLSOrKwspaWlac6cOcrNzZXVatWoUaNUXFzcsjeGq06PHj00b9485eTkaMeOHUpISNBdd92lr7/+usH+fJevPNu3b9eiRYsUExPTZD8+e3iUAbcrLi42JBkbNmxw2FZXV2dYrVbjN7/5jVFbW2tr37dvnxEaGmo8//zzFz2+JGPFihUO7YMHDzZSU1PtxgoLCzPS09MNwzCMzZs3G8nJybbtU6dONd5++21n3hquQh07djTeeOMNh3a+y1eeM2fOGH369DHWrFljxMfHG1OnTm2wH589PI0zM5dBaWmpJCkkJMRhm5eXl1auXKmdO3dqwoQJqq+v18GDB5WQkKDk5GTNmDGjRWOeO3dOOTk5SkxMtBsrMTFRW7ZskSQNHjxYX331lf7xj3+ovLxcn3zyiUaNGtWi8XDlq6urU2ZmpioqKjRs2DCH7XyXrzypqam6/fbb7f7dN4TPHp7m4+kCrnT19fV64oknNHz4cN10000N9gkLC9PatWsVFxen8ePHa8uWLUpMTNTChQtbPO6JEydUV1en0NBQu/bQ0FDt27dPkuTj46M//elPGjFihOrr6zVjxgxWAMDBnj17NGzYMFVXVyswMFArVqxQdHR0g335Ll85MjMzlZubq+3btzerP589PIkw42apqan66quvtGnTpib7RUREaNmyZYqPj1evXr20ePFiWSwWt9d355136s4773T7ODCv66+/Xnl5eSotLdXy5cuVkpKiDRs2NBpo+C6b35EjRzR16lStWbNG/v7+zd6Pzx6ewmUmN5oyZYo+/vhjrVu3Tj169Giyb1FRkSZPnqykpCRVVlZq2rRplzR2586d5e3t7TDxrqioSF27dr2kY+Pq4uvrq969e2vAgAFKT0+X1WrVyy+/3Gh/vsvml5OTo+LiYvXv318+Pj7y8fHRhg0b9Morr8jHx0d1dXUN7sdnD08hzLiBYRiaMmWKVqxYobVr1+raa69tsv+JEyc0cuRI9e3bV++9956ys7OVlZWl6dOnt7gGX19fDRgwQNnZ2ba2+vp6ZWdnNzjfAWiu+vp6nT17tsFtfJevDCNHjtSePXuUl5dn+xs4cKDuv/9+5eXlydvb22EfPnt4lKdnIF+JHnvsMSM4ONhYv369UVhYaPurrKx06FtXV2cMHDjQuO2224yzZ8/a2vPy8oyQkBDjpZdeanCMM2fOGDt37jR27txpSDJeeuklY+fOncYPP/xg65OZmWn4+fkZS5cuNfbu3WtMnjzZ6NChg3Hs2DHXv2lckWbOnGls2LDByM/PN3bv3m3MnDnTsFgsxurVqx368l2+sl1sNROfPTyJMOMGkhr8e/PNNxvsv3r1aqOqqsqhPTc31zhy5EiD+6xbt67BMVJSUuz6vfrqq0ZERITh6+trDB482Ni6deulvj1cRR588EEjMjLS8PX1Nbp06WKMHDmywSBzHt/lK1dTYcYw+OzhWTybCQAAmBpzZgAAgKkRZgAAgKkRZgAAgKkRZgAAgKkRZgAAgKkRZgAAgKkRZgAAgKkRZgAAgKkRZgAAgKkRZgBcEW655RY98cQTHhv/F7/4hf7617+67firVq1SbGys6uvr3TYGYFaEGQAuMXHiRFksFlksFrVp00bXXnutZsyYoerqapeOs379elksFp0+fdqu/b333tMf/vAHl47VXB9++KGKioo0btw4t40xevRotWnTRm+//bbbxgDMijADwGVGjx6twsJCff/995o/f74WLVqkOXPmXJaxQ0JC1L59+8sy1s+98sormjRpkry83Puf1IkTJ+qVV15x6xiAGRFmALiMn5+funbtqvDwcCUnJysxMVFr1qyxbe/Zs6cWLFhgt09sbKyefvpp22uLxaI33nhDv/rVr9SuXTv16dNHH374oSTp0KFDGjFihCSpY8eOslgsmjhxoiTHy0w9e/bUs88+qwkTJigwMFCRkZH68MMPdfz4cd11110KDAxUTEyMduzYYVfPpk2bFBcXp7Zt2yo8PFyPP/64KioqGn3Px48f19q1a5WUlGTXbrFYtGjRIt1xxx1q166d+vbtqy1btui7777TLbfcooCAAP3rv/6rDh48aNtn165dGjFihNq3b6+goCANGDDArr6kpCTt2LHDbh8AhBkAbvLVV1/piy++kK+vr9P7PvPMM7r33nu1e/du3Xbbbbr//vtVUlKi8PBwvfvuu5Kk/fv3q7CwUC+//HKjx5k/f76GDx+unTt36vbbb9dvf/tbTZgwQQ888IByc3N13XXXacKECTIMQ5J08OBBjR49Wnfffbd2796trKwsbdq0SVOmTGl0jE2bNtnCys/94Q9/0IQJE5SXl6cbbrhB48eP16OPPqpZs2Zpx44dMgzD7tj333+/evTooe3btysnJ0czZ85UmzZtbNsjIiIUGhqqzz//3Ol/p8AVzQAAF0hJSTG8vb2NgIAAw8/Pz5BkeHl5GcuXL7f1iYyMNObPn2+3n9VqNebMmWN7Lcl48sknba/Ly8sNScYnn3xiGIZhrFu3zpBknDp1yu448fHxxtSpU+3GeuCBB2yvCwsLDUnGU089ZWvbsmWLIckoLCw0DMMwHnroIWPy5Ml2x/38888NLy8vo6qqqsH3PX/+fKNXr14O7T9/H+fHWrx4sa3tb3/7m+Hv72973b59e2Pp0qUNjnNev379jKeffrrJPsDVhjMzAFxmxIgRysvL05dffqmUlBRNmjRJd999t9PHiYmJsf1zQECAgoKCVFxcfEnHCQ0NlST9y7/8i0Pb+WPv2rVLS5cuVWBgoO1v1KhRqq+vV35+foNjVFVVyd/fv8XjV1dXq6ysTJKUlpamhx9+WImJiZo3b16Dl5Patm2rysrKi7954CpCmAHgMgEBAerdu7esVquWLFmiL7/8UosXL7Zt9/Lysl3SOa+mpsbhOBdeWpF+nH/SkiXJFx7HYrE02nb+2OXl5Xr00UeVl5dn+9u1a5cOHDig6667rsExOnfurFOnTrlk/Kefflpff/21br/9dq1du1bR0dFasWKF3TFLSkrUpUuXZrx74OpBmAHgFl5eXvr973+vJ598UlVVVZKkLl26qLCw0NanrKys0TMejTk/B6eurs51xf5T//79tXfvXvXu3dvhr7G5P/369dOxY8caDTTOioqK0rRp07R69Wr9+te/1ptvvmnbVl1drYMHD6pfv34uGQu4UhBmALjNPffcI29vb2VkZEiSEhIStGzZMn3++efas2ePUlJS5O3t7dQxIyMjZbFY9PHHH+v48eMqLy93Wb2/+93v9MUXX2jKlCnKy8vTgQMH9MEHHzQ5Abhfv37q3LmzNm/efEljV1VVacqUKVq/fr1++OEHbd68Wdu3b7ebWLx161b5+flp2LBhlzQWcKUhzABwGx8fH02ZMkV//OMfVVFRoVmzZik+Pl533HGHbr/9diUnJzd6+aYx3bt31zPPPKOZM2cqNDS0yaDhrJiYGG3YsEHffvut4uLi1K9fP82ePVthYWGN7uPt7a1JkyZd8s3svL29dfLkSU2YMEFRUVG69957NWbMGD3zzDO2Pn/72990//33q127dpc0FnClsRg/v4ANAHDKsWPHdOONNyo3N1eRkZFuGePEiRO6/vrrtWPHDl177bVuGQMwK87MAMAl6tq1qxYvXqzDhw+7bYxDhw7pz3/+M0EGaABnZgAAgKlxZgYAAJgaYQYAAJgaYQYAAJgaYQYAAJgaYQYAAJgaYQYAAJgaYQYAAJgaYQYAAJgaYQYAAJja/wMHczG9fghGPgAAAABJRU5ErkJggg==", |
| 972 | "text/plain": [ |
| 973 | "<Figure size 600x400 with 1 Axes>" |
| 974 | ] |
| 975 | }, |
| 976 | "execution_count": 10, |
| 977 | "metadata": {}, |
| 978 | "output_type": "execute_result" |
| 979 | } |
| 980 | ], |
| 981 | "source": [ |
| 982 | "# Log-log Pareto frontier: runtime (x) vs physical qubits (y)\n", |
| 983 | "plot_estimates(results, figsize=(6, 4), runtime_unit=\"ms\")" |
| 984 | ] |
| 985 | }, |
| 986 | { |
| 987 | "cell_type": "markdown", |
| 988 | "id": "ebb0739c", |
| 989 | "metadata": {}, |
| 990 | "source": [ |
| 991 | "The plot visualizes the Pareto frontier: each point represents a configuration that is optimal in the sense that no other configuration achieves both fewer qubits *and* a shorter runtime. Moving left along the x-axis trades qubits for speed, while moving down trades speed for fewer qubits." |
| 992 | ] |
| 993 | }, |
| 994 | { |
| 995 | "cell_type": "markdown", |
| 996 | "id": "ab667cbc", |
| 997 | "metadata": {}, |
| 998 | "source": [ |
| 999 | "## Compare multiple estimations\n", |
| 1000 | "\n", |
| 1001 | "A common workflow is to compare resource estimates across different hardware assumptions. Here we sweep over three physical error rates and run a separate estimation for each. The `name` parameter labels each run so the results can be distinguished in the plot. Passing the list of `EstimationTable` objects to `plot_estimates` overlays them on a single figure with a legend." |
| 1002 | ] |
| 1003 | }, |
| 1004 | { |
| 1005 | "cell_type": "code", |
| 1006 | "execution_count": 11, |
| 1007 | "id": "c2e31f33", |
| 1008 | "metadata": {}, |
| 1009 | "outputs": [ |
| 1010 | { |
| 1011 | "data": { |
| 1012 | "image/png": "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tbW1euutt3TkyBFJUmpqqn7yk584LYftKBjZAACgdVr6Hep2Ovjkk0/04x//WBaLRf369ZMkPf300+rWrZveeecdXX311a2vGgAA+B23O4jed999GjBggE6dOqUDBw7owIEDOnnypAYNGqTp06cbUSMAAOjA3B7ZKCgo0L59+9SlSxfHvi5dumjJkiXNtgkHAACBye2Rjb59+zbq6ClJJSUl6tOnj0eKAgAA/qNFYaO8vNzxs3TpUs2ePVubNm3SqVOndOrUKW3atEkPP/ywnn76aaPrBQAAHUyLVqMEBQU5NfKyf8S+r+F2bW2tEXUahtUoAAC0jkdXo/z973/3WGEAACCwtChsjBo1yug6AACAn3J7Ncr27dubPf7DH/6w1cW0p4YdRAEAgHHc7iAaFNR4TmnD+Rwd7cubORsAALROS79D3V76evbsWaefkpISbd68WcOGDdOWLVvaVDQAAPA/bj9GMZvNjfb96Ec/UlhYmHJycrR//36PFAYAAPyD2yMbrsTHx+vTTz/11OUAAICfcHtk4+OPP3battlsKioq0q9+9SulpaV5qi4AAOAn3A4baWlpMplMunRe6ciRI7VmzRqPFQYAAPyD22GjsLDQaTsoKEjdunVTRESEx4oCAAD+w+2wkZycbEQdAADAT7kdNp5//vkWnzt79mx3Lw8AAPyM2029evfurTNnzujcuXPq3LmzJOmbb75RZGSkunXr9u2FTSYdO3bMo8V6UsMOop999hlNvQAAcJNhTb2WLFmitLQ0HTlyRKWlpSotLdWRI0d0zTXX6Mknn1RhYaEKCwt9OmhI0syZM3X48GF99NFH3i4FAAC/5vbIxhVXXKFNmzZpyJAhTvv379+vn/70p40mkPo62pUDANA6ho1sFBUV6eLFi43219bWqri42N3LAQAAP+d22Ljxxhs1Y8YMHThwwLFv//79evDBB5WRkeHR4gAAQMfndthYs2aNEhISNHToUIWHhys8PFzDhw9XfHy8Vq9ebUSNAACgA3N76Wu3bt307rvv6vPPP9eRI0ckSf3791ffvn09XhwAAOj43A4bdldeeaWuvPJKT9YCAAD8kMfe+goAANAUwgYAADAUYQMAABiKsAEAAAzVogmiH3/8cYsvOGjQoFYXAwAA/E+LwkZaWppMJpNcdTa3HzOZTKqtrfVogUZp+CI2AABgnBa9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//79NXXqVM2YMUPz5s3Tvn37ZLPZnK595513KjExUR999JH279+vuXPnKjQ01HG8V69eio+PV35+vtt/poBfswEICFlZWbbg4GBbVFSULTw83CbJFhQUZNu0aZPjnOTkZFtubq7T5wYPHmxbtGiRY1uSbf78+Y7tyspKmyTbX//6V5vNZrP9/e9/t0mynT171uk6o0aNsj300ENO97rrrrsc20VFRTZJtgULFjj27dq1yybJVlRUZLPZbLZ7773XNn36dKfr5ufn24KCgmxVVVVN/t65ubm2733ve432X/p72O/1yiuvOPb97ne/s0VERDi2Y2JibOvWrWvyPnZDhgyxPfbYY82eAwQaRjaAADJmzBgVFBRoz549ysrKUnZ2tiZNmuT2dQYNGuT4z1FRUYqNjVVJSUmbrhMfHy9JGjhwYKN99msfPHhQ69atU3R0tONn7NixqqurU2FhYZP3qKqqUkRERKvvf/78eZWXl0uScnJydN999ykjI0O/+tWvmnxc0qlTJ507d+67f3kggBA2gAASFRWlPn36aPDgwVqzZo327NmjV155xXE8KCjI8cjC7sKFC42u0/DRgVQ//6E1Sz4bXsdkMrncZ792ZWWlZsyYoYKCAsfPwYMH9fnnn+uKK65o8h5du3bV2bNnPXL/xx57TJ988okmTJigDz74QKmpqXrrrbecrllaWqpu3bq14LcHAgdhAwhQQUFB+uUvf6n58+erqqpKktStWzcVFRU5zikvL3c5YuCKfQ5IbW2t54r9f9dcc40OHz6sPn36NPpxNfdkyJAhslgsLgOHu/r27atHHnlEW7Zs0e233661a9c6jp0/f15Hjx7VkCFDPHIvwF8QNoAAdscddyg4OFh5eXmSpBtuuEGvvfaa8vPzdejQIWVlZSk4ONitayYnJ8tkMukvf/mLzpw5o8rKSo/V+1//9V/auXOnZs2apYKCAn3++ef605/+1OwE0SFDhqhr16768MMP23TvqqoqzZo1S9u2bdOXX36pDz/8UB999JHTxNPdu3crPDxc6enpbboX4G8IG0AACwkJ0axZs/TMM8/IarVq3rx5GjVqlG655RZNmDBBt956q8vHE6707NlTixcv1ty5cxUfH99sEHDXoEGD9L//+7/67LPPdP3112vIkCFauHChevTo4fIzwcHBys7ObnOzreDgYP373/9WZmam+vbtq5/97GcaP368Fi9e7Djnd7/7ne68805FRka26V6AvzHZLn1ACwB+xmKxaMCAATpw4ICSk5MNucfXX3+tfv36ad++ferdu7ch9wA6KkY2APi9hIQEvfLKKzpx4oRh9zh+/LheeuklggbQBEY2AACAoRjZAAAAhiJsAAAAQxE2AACAoQgbAADAUIQNAABgKMIGAAAwFGEDAAAYirABAAAMRdgAAACG+j9hBkvUTmIu9gAAAABJRU5ErkJggg==", |
| 1013 | "text/plain": [ |
| 1014 | "<Figure size 600x400 with 1 Axes>" |
| 1015 | ] |
| 1016 | }, |
| 1017 | "execution_count": 11, |
| 1018 | "metadata": {}, |
| 1019 | "output_type": "execute_result" |
| 1020 | } |
| 1021 | ], |
| 1022 | "source": [ |
| 1023 | "# Compare estimates across different physical error rates\n", |
| 1024 | "results = []\n", |
| 1025 | "\n", |
| 1026 | "for error_rate in [1e-3, 1e-4, 1e-5]:\n", |
| 1027 | " arch = GateBased(error_rate=error_rate, gate_time=100, measurement_time=500)\n", |
| 1028 | " results.append(estimate(app, arch, isa_query=SurfaceCode.q() * RoundBasedFactory.q(), max_error=0.01, name=f\"p = {error_rate:.0e}\"))\n", |
| 1029 | "\n", |
| 1030 | "# Overlay multiple estimation runs on a single plot\n", |
| 1031 | "plot_estimates(results, figsize=(6, 4), runtime_unit=\"ms\")" |
| 1032 | ] |
| 1033 | }, |
| 1034 | { |
| 1035 | "cell_type": "markdown", |
| 1036 | "id": "0bb282a2", |
| 1037 | "metadata": {}, |
| 1038 | "source": [ |
| 1039 | "## Next steps\n", |
| 1040 | "\n", |
| 1041 | "This notebook covered the basic workflow for quantum resource estimation. To explore further:\n", |
| 1042 | "\n", |
| 1043 | "- Try different Q# applications from the `samples/` directory\n", |
| 1044 | "- Adjust the architecture parameters (e.g., different `error_rate` or `gate_time` values) to see how hardware assumptions affect the estimates\n", |
| 1045 | "- Explore different ISA queries, such as varying the surface code distance range with `SurfaceCode.q(distance=range(3, 30, 2))`\n", |
| 1046 | "- Compare results across multiple estimation runs using `plot_estimates`\n", |
| 1047 | "\n", |
| 1048 | "To learn how to estimate quantum programs written in various languages and frameworks, check out the [Importing quantum programs into QRE](1_qre_input.ipynb) notebook." |
| 1049 | ] |
| 1050 | } |
| 1051 | ], |
| 1052 | "metadata": { |
| 1053 | "kernelspec": { |
| 1054 | "display_name": ".venv", |
| 1055 | "language": "python", |
| 1056 | "name": "python3" |
| 1057 | }, |
| 1058 | "language_info": { |
| 1059 | "codemirror_mode": { |
| 1060 | "name": "ipython", |
| 1061 | "version": 3 |
| 1062 | }, |
| 1063 | "file_extension": ".py", |
| 1064 | "mimetype": "text/x-python", |
| 1065 | "name": "python", |
| 1066 | "nbconvert_exporter": "python", |
| 1067 | "pygments_lexer": "ipython3", |
| 1068 | "version": "3.11.14" |
| 1069 | } |
| 1070 | }, |
| 1071 | "nbformat": 4, |
| 1072 | "nbformat_minor": 5 |
| 1073 | } |
| 1074 | |