From 434499225ba1db0e98d9835019440aa02cb99a33 Mon Sep 17 00:00:00 2001 From: Damian Rovara <93778306+DRovara@users.noreply.github.com> Date: Wed, 25 Sep 2024 16:50:13 +0200 Subject: [PATCH] :memo: Add Documentation (#24) * docs: :memo: Add RTD docs * docs: :memo: Write library documentation * docs: :recycle: Align "MQT Debug" vs "MQT Debugger" to "MQT Debugger" * docs: :memo: Add docstrings to python bindings * docs: :memo: Add interactive debugging section to docs and add docstrings to python bindings * fix: :pencil2: Fix links to QCEC instead of mqt debugger * fix: :lipstick: Fix linter issues * refactor: :label: Ann py.typed marker and pyi file for _version * fix: :construction_worker: Remove CMAKE_GENERATOR = Ninja from windows cibuildwheel tool --- .github/ISSUE_TEMPLATE/bug-report.yml | 4 +- .github/ISSUE_TEMPLATE/feature-request.yml | 2 +- .github/codecov.yml | 4 +- .github/contributing.rst | 16 +- .github/release-drafter.yml | 2 +- .github/support.rst | 6 +- .github/workflows/cd.yml | 2 +- CMakeLists.txt | 21 +- README.md | 32 +- app/CMakeLists.txt | 12 +- app/testDDSimDebugger.cpp | 16 +- cmake/ExternalDependencies.cmake | 6 +- docs/.gitignore | 2 + docs/Doxyfile | 143 +++ docs/source/Assertions.rst | 116 +++ docs/source/Contributing.rst | 1 + docs/source/Debugging.rst | 88 ++ docs/source/DevelopmentGuide.rst | 59 ++ docs/source/Diagnosis.rst | 86 ++ docs/source/Installation.rst | 96 ++ docs/source/Publications.rst | 14 + docs/source/Quickstart.rst | 100 +++ docs/source/Support.rst | 1 + docs/source/_static/custom.css | 40 + docs/source/_static/mqt_dark.png | Bin 0 -> 61137 bytes docs/source/_static/mqt_light.png | Bin 0 -> 57266 bytes docs/source/_templates/page.html | 76 ++ docs/source/conf.py | 153 ++++ docs/source/index.rst | 50 ++ docs/source/library/Library.rst | 10 + docs/source/library/dd/DDSimDebug.rst | 5 + docs/source/library/dd/DDSimDiagnostics.rst | 5 + docs/source/library/dd/Dd.rst | 11 + docs/source/library/interface/Common.rst | 5 + docs/source/library/interface/Debug.rst | 10 + docs/source/library/interface/Diagnostics.rst | 10 + docs/source/library/interface/Interface.rst | 16 + .../library/parsing/AssertionParsing.rst | 5 + .../library/parsing/CodePreprocessing.rst | 5 + docs/source/library/parsing/Parsing.rst | 13 + docs/source/library/parsing/ParsingError.rst | 5 + docs/source/library/parsing/Utils.rst | 5 + docs/source/library/python/Debug.rst | 5 + docs/source/library/python/Python.rst | 10 + docs/source/library/python/dap/Adapter.rst | 16 + docs/source/library/python/dap/DAP.rst | 13 + docs/source/library/python/dap/Messages.rst | 7 + docs/source/refs.bib | 0 include/backend/dd/DDSimDebug.hpp | 6 +- include/backend/debug.h | 21 +- include/backend/diagnostics.h | 15 +- include/common.h | 1 - pyproject.toml | 56 +- src/CMakeLists.txt | 4 +- src/backend/dd/DDSimDebug.cpp | 4 +- src/mqt/debug/pydebug.pyi | 98 --- src/mqt/{debug => debugger}/__init__.py | 8 +- src/mqt/debugger/_version.pyi | 4 + src/mqt/{debug => debugger}/dap/__init__.py | 0 src/mqt/{debug => debugger}/dap/adapter.py | 0 src/mqt/{debug => debugger}/dap/dap_server.py | 89 +- .../dap/messages/__init__.py | 0 .../dap/messages/capabilities_dap_event.py | 0 .../configuration_done_dap_message.py | 0 .../dap/messages/continue_dap_message.py | 0 .../dap/messages/dap_event.py | 0 .../dap/messages/dap_message.py | 0 .../dap/messages/disconnect_dap_message.py | 4 +- .../dap/messages/exception_info_message.py | 0 .../dap/messages/exited_dap_event.py | 0 .../dap/messages/gray_out_event.py | 0 .../dap/messages/initialize_dap_message.py | 4 +- .../dap/messages/initialized_dap_event.py | 0 .../dap/messages/launch_dap_message.py | 0 .../dap/messages/next_dap_message.py | 0 .../dap/messages/output_dap_event.py | 0 .../dap/messages/pause_dap_message.py | 0 .../dap/messages/restart_dap_message.py | 0 .../dap/messages/restart_frame_dap_message.py | 0 .../messages/reverse_continue_dap_message.py | 0 .../dap/messages/scopes_dap_message.py | 0 .../messages/set_breakpoints_dap_message.py | 0 .../set_exception_breakpoints_dap_message.py | 0 .../dap/messages/stack_trace_dap_message.py | 0 .../dap/messages/step_back_dap_message.py | 0 .../dap/messages/step_in_dap_message.py | 0 .../dap/messages/step_out_dap_message.py | 0 .../dap/messages/stopped_dap_event.py | 0 .../dap/messages/terminate_dap_message.py | 4 +- .../dap/messages/terminated_dap_event.py | 0 .../dap/messages/threads_dap_message.py | 0 .../{debug => debugger}/dap/messages/utils.py | 0 .../dap/messages/variables_dap_message.py | 0 src/mqt/debugger/py.typed | 0 src/mqt/debugger/pydebugger.pyi | 493 +++++++++++ src/python/CMakeLists.txt | 10 +- src/python/InterfaceBindings.cpp | 819 +++++++++++++----- src/python/bindings.cpp | 4 +- src/python/dd/DDSimDebugBindings.cpp | 34 +- test/CMakeLists.txt | 6 +- test/python/test_diagnosis.py | 2 +- test/python/test_end_to_end.py | 4 +- test/python/test_python_bindings.py | 26 +- 103 files changed, 2481 insertions(+), 539 deletions(-) create mode 100644 docs/.gitignore create mode 100644 docs/Doxyfile create mode 100644 docs/source/Assertions.rst create mode 100644 docs/source/Contributing.rst create mode 100644 docs/source/Debugging.rst create mode 100644 docs/source/DevelopmentGuide.rst create mode 100644 docs/source/Diagnosis.rst create mode 100644 docs/source/Installation.rst create mode 100644 docs/source/Publications.rst create mode 100644 docs/source/Quickstart.rst create mode 100644 docs/source/Support.rst create mode 100644 docs/source/_static/custom.css create mode 100644 docs/source/_static/mqt_dark.png create mode 100644 docs/source/_static/mqt_light.png create mode 100644 docs/source/_templates/page.html create mode 100644 docs/source/conf.py create mode 100644 docs/source/index.rst create mode 100644 docs/source/library/Library.rst create mode 100644 docs/source/library/dd/DDSimDebug.rst create mode 100644 docs/source/library/dd/DDSimDiagnostics.rst create mode 100644 docs/source/library/dd/Dd.rst create mode 100644 docs/source/library/interface/Common.rst create mode 100644 docs/source/library/interface/Debug.rst create mode 100644 docs/source/library/interface/Diagnostics.rst create mode 100644 docs/source/library/interface/Interface.rst create mode 100644 docs/source/library/parsing/AssertionParsing.rst create mode 100644 docs/source/library/parsing/CodePreprocessing.rst create mode 100644 docs/source/library/parsing/Parsing.rst create mode 100644 docs/source/library/parsing/ParsingError.rst create mode 100644 docs/source/library/parsing/Utils.rst create mode 100644 docs/source/library/python/Debug.rst create mode 100644 docs/source/library/python/Python.rst create mode 100644 docs/source/library/python/dap/Adapter.rst create mode 100644 docs/source/library/python/dap/DAP.rst create mode 100644 docs/source/library/python/dap/Messages.rst create mode 100644 docs/source/refs.bib delete mode 100644 src/mqt/debug/pydebug.pyi rename src/mqt/{debug => debugger}/__init__.py (73%) create mode 100644 src/mqt/debugger/_version.pyi rename src/mqt/{debug => debugger}/dap/__init__.py (100%) rename src/mqt/{debug => debugger}/dap/adapter.py (100%) rename src/mqt/{debug => debugger}/dap/dap_server.py (81%) rename src/mqt/{debug => debugger}/dap/messages/__init__.py (100%) rename src/mqt/{debug => debugger}/dap/messages/capabilities_dap_event.py (100%) rename src/mqt/{debug => debugger}/dap/messages/configuration_done_dap_message.py (100%) rename src/mqt/{debug => debugger}/dap/messages/continue_dap_message.py (100%) rename src/mqt/{debug => debugger}/dap/messages/dap_event.py (100%) rename src/mqt/{debug => debugger}/dap/messages/dap_message.py (100%) rename src/mqt/{debug => debugger}/dap/messages/disconnect_dap_message.py (91%) rename src/mqt/{debug => debugger}/dap/messages/exception_info_message.py (100%) rename src/mqt/{debug => debugger}/dap/messages/exited_dap_event.py (100%) rename src/mqt/{debug => debugger}/dap/messages/gray_out_event.py (100%) rename src/mqt/{debug => debugger}/dap/messages/initialize_dap_message.py (95%) rename src/mqt/{debug => debugger}/dap/messages/initialized_dap_event.py (100%) rename src/mqt/{debug => debugger}/dap/messages/launch_dap_message.py (100%) rename src/mqt/{debug => debugger}/dap/messages/next_dap_message.py (100%) rename src/mqt/{debug => debugger}/dap/messages/output_dap_event.py (100%) rename src/mqt/{debug => debugger}/dap/messages/pause_dap_message.py (100%) rename src/mqt/{debug => debugger}/dap/messages/restart_dap_message.py (100%) rename src/mqt/{debug => debugger}/dap/messages/restart_frame_dap_message.py (100%) rename src/mqt/{debug => debugger}/dap/messages/reverse_continue_dap_message.py (100%) rename src/mqt/{debug => debugger}/dap/messages/scopes_dap_message.py (100%) rename src/mqt/{debug => debugger}/dap/messages/set_breakpoints_dap_message.py (100%) rename src/mqt/{debug => debugger}/dap/messages/set_exception_breakpoints_dap_message.py (100%) rename src/mqt/{debug => debugger}/dap/messages/stack_trace_dap_message.py (100%) rename src/mqt/{debug => debugger}/dap/messages/step_back_dap_message.py (100%) rename src/mqt/{debug => debugger}/dap/messages/step_in_dap_message.py (100%) rename src/mqt/{debug => debugger}/dap/messages/step_out_dap_message.py (100%) rename src/mqt/{debug => debugger}/dap/messages/stopped_dap_event.py (100%) rename src/mqt/{debug => debugger}/dap/messages/terminate_dap_message.py (91%) rename src/mqt/{debug => debugger}/dap/messages/terminated_dap_event.py (100%) rename src/mqt/{debug => debugger}/dap/messages/threads_dap_message.py (100%) rename src/mqt/{debug => debugger}/dap/messages/utils.py (100%) rename src/mqt/{debug => debugger}/dap/messages/variables_dap_message.py (100%) create mode 100644 src/mqt/debugger/py.typed create mode 100644 src/mqt/debugger/pydebugger.pyi diff --git a/.github/ISSUE_TEMPLATE/bug-report.yml b/.github/ISSUE_TEMPLATE/bug-report.yml index c9b57a5..b56a717 100644 --- a/.github/ISSUE_TEMPLATE/bug-report.yml +++ b/.github/ISSUE_TEMPLATE/bug-report.yml @@ -8,9 +8,9 @@ body: **Thank you for wanting to report a bug for this project!** ⚠ - Verify first that your issue is not [already reported on GitHub](https://github.com/cda-tum/qcec/search?q=is%3Aissue&type=issues). + Verify first that your issue is not [already reported on GitHub](https://github.com/cda-tum/mqt-debugger/search?q=is%3Aissue&type=issues). - If you have general questions, please consider [starting a discussion](https://github.com/cda-tum/qcec/discussions). + If you have general questions, please consider [starting a discussion](https://github.com/cda-tum/mqt-debugger/discussions). - type: textarea attributes: label: Environment information diff --git a/.github/ISSUE_TEMPLATE/feature-request.yml b/.github/ISSUE_TEMPLATE/feature-request.yml index 812010c..e5d8da8 100644 --- a/.github/ISSUE_TEMPLATE/feature-request.yml +++ b/.github/ISSUE_TEMPLATE/feature-request.yml @@ -8,7 +8,7 @@ body: **Thank you for wanting to suggest a feature for this project!** ⚠ - Verify first that your idea is not [already requested on GitHub](https://github.com/cda-tum/qcec/search?q=is%3Aissue&type=issues). + Verify first that your idea is not [already requested on GitHub](https://github.com/cda-tum/mqt-debugger/search?q=is%3Aissue&type=issues). - type: textarea attributes: diff --git a/.github/codecov.yml b/.github/codecov.yml index 20f3769..61b40c8 100644 --- a/.github/codecov.yml +++ b/.github/codecov.yml @@ -2,8 +2,8 @@ ignore: - "extern/**/*" - "**/python" - "test/**/*" - - "src/mqt/debug/dap/**/*" - - "src/mqt/debug/dap/*" + - "src/mqt/debugger/dap/**/*" + - "src/mqt/debugger/dap/*" - "app/*" - "src/frontend/**/*" diff --git a/.github/contributing.rst b/.github/contributing.rst index 7704d3c..99b62a7 100644 --- a/.github/contributing.rst +++ b/.github/contributing.rst @@ -6,7 +6,7 @@ We value contributions from people with all levels of experience. In particular if this is your first pull request not everything has to be perfect. We will guide you through the process. -We use GitHub to `host code `_, to `track issues and feature requests `_, as well as accept `pull requests `_. +We use GitHub to `host code `_, to `track issues and feature requests `_, as well as accept `pull requests `_. See https://docs.github.com/en/get-started/quickstart for a general introduction to working with GitHub and contributing to projects. Types of Contributions @@ -15,24 +15,24 @@ Types of Contributions You can contribute in several ways: - 🐛 Report Bugs - Report bugs at https://github.com/cda-tum/mqt-debug/issues using the *🐛 Bug report* issue template. Please make sure to fill out all relevant information in the respective issue form. + Report bugs at https://github.com/cda-tum/mqt-debugger/issues using the *🐛 Bug report* issue template. Please make sure to fill out all relevant information in the respective issue form. - 🐛 Fix Bugs - Look through the `GitHub Issues `_ for bugs. Anything tagged with "bug" is open to whoever wants to try and fix it. + Look through the `GitHub Issues `_ for bugs. Anything tagged with "bug" is open to whoever wants to try and fix it. - ✨ Propose New Features - Propose new features at https://github.com/cda-tum/mqt-debug/issues using the *✨ Feature request* issue template. Please make sure to fill out all relevant information in the respective issue form. + Propose new features at https://github.com/cda-tum/mqt-debugger/issues using the *✨ Feature request* issue template. Please make sure to fill out all relevant information in the respective issue form. - ✨ Implement New Features - Look through the `GitHub Issues `_ for features. Anything tagged with "feature" is open to whoever wants to implement it. We highly appreciate external contributions to the project. + Look through the `GitHub Issues `_ for features. Anything tagged with "feature" is open to whoever wants to implement it. We highly appreciate external contributions to the project. - 📝 Write Documentation - MQT Debug could always use some more `documentation `_, and we appreciate any help with that. + MQT Debugger could always use some more `documentation `_, and we appreciate any help with that. 🎉 Get Started ############## -Ready to contribute? Check out the :doc:`Development Guide ` to set up MQT Debug for local development and learn about the style guidelines and conventions used throughout the project. +Ready to contribute? Check out the :doc:`Development Guide ` to set up MQT Debugger for local development and learn about the style guidelines and conventions used throughout the project. We value contributions from people with all levels of experience. In particular if this is your first PR not everything has to be perfect. @@ -62,7 +62,7 @@ Pull Request Workflow - If any of the :code:`Python Packaging/\*` checks fail, this indicates an error in the Python bindings or creation of the Python wheels and/or source distribution. Look through the respective logs on GitHub for any error or failure messages. - If any of the :code:`Python/\*` checks fail, this indicates an error in the Python part of the code base. Look through the respective logs on GitHub for any error or failure messages. - If any of the :code:`codecov/\*` checks fail, this means that your changes are not appropriately covered by tests or that the overall project coverage decreased too much. Ensure that you include tests for all your changes in the PR. - - If the :code:`docs/readthedocs.org:debug` check fails, the documentation could not be built properly. Inspect the corresponding log file for any errors. + - If the :code:`docs/readthedocs.org:debugger` check fails, the documentation could not be built properly. Inspect the corresponding log file for any errors. - If :code:`cpp-linter` comments on your PR with a list of warnings, these have been raised by :code:`clang-tidy` when checking the C++ part of your changes for warnings or style guideline violations. The individual messages frequently provide helpful suggestions on how to fix the warnings. - If the :code:`pre-commit.ci` check fails, some of the :code:`pre-commit` checks failed and could not be fixed automatically by the *pre-commit.ci* bot. Such failures are most likely related to the Python part of the code base. The individual log messages frequently provide helpful suggestions on how to fix the warnings. diff --git a/.github/release-drafter.yml b/.github/release-drafter.yml index 8ec9986..03ef4c0 100644 --- a/.github/release-drafter.yml +++ b/.github/release-drafter.yml @@ -1,4 +1,4 @@ -name-template: "MQT Debug $RESOLVED_VERSION Release" +name-template: "MQT Debugger $RESOLVED_VERSION Release" tag-template: "v$RESOLVED_VERSION" categories: - title: "🚀 Features and Enhancements" diff --git a/.github/support.rst b/.github/support.rst index 1d8dc2b..b4bdc3f 100644 --- a/.github/support.rst +++ b/.github/support.rst @@ -1,12 +1,12 @@ Support ======= -If you are stuck with a problem using MQT Debug or are having questions, please do get in touch at our `Issues `_ or `Discussions `_. We'd love to help. +If you are stuck with a problem using MQT Debugger or are having questions, please do get in touch at our `Issues `_ or `Discussions `_. We'd love to help. You can save time by following this procedure when reporting a problem: -- Do try to solve the problem on your own first. Make sure to consult the `Documentation `_. -- Search through past `Issues `_ to see if someone else already had the same problem. +- Do try to solve the problem on your own first. Make sure to consult the `Documentation `_. +- Search through past `Issues `_ to see if someone else already had the same problem. - Before filing a bug report, try to create a minimal working example (MWE) that reproduces the problem. It's much easier to identify the cause for the problem if a handful of lines suffice to show that something isn't working. You can also always reach us at `quantum.cda@xcit.tum.de `_. diff --git a/.github/workflows/cd.yml b/.github/workflows/cd.yml index 281f402..7d76acb 100644 --- a/.github/workflows/cd.yml +++ b/.github/workflows/cd.yml @@ -18,7 +18,7 @@ jobs: runs-on: ubuntu-latest environment: name: pypi - url: https://pypi.org/p/mqt.qcec + url: https://pypi.org/p/mqt.debugger permissions: id-token: write attestations: write diff --git a/CMakeLists.txt b/CMakeLists.txt index 944dbf1..42aba74 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -1,15 +1,16 @@ cmake_minimum_required(VERSION 3.26) project( - mqt_debug + mqt_debugger LANGUAGES CXX - DESCRIPTION "MQT Debug - A debugging tool for Quantum Circuits") + DESCRIPTION "MQT Debugger - A debugging tool for Quantum Circuits") -option(BUILD_MQT_DEBUG_BINDINGS "Build the MQT DEBUG Python bindings" OFF) -option(BUILD_MQT_DEBUG_TESTS "Also build tests for the MQT QCEC project" ON) +option(BUILD_MQT_DEBUGGER_BINDINGS "Build the MQT Debugger Python bindings" OFF) +option(BUILD_MQT_DEBUGGER_TESTS "Also build tests for the MQT Debugger project" ON) +option(BUILD_MQT_DEBUGGER_APP "Also build the CLI app for the MQT Debugger project" ON) set(CMAKE_CXX_STANDARD 17) -if(BUILD_MQT_DEBUG_BINDINGS) +if(BUILD_MQT_DEBUGGER_BINDINGS) # ensure that the BINDINGS option is set set(BINDINGS ON @@ -41,10 +42,12 @@ include(cmake/ExternalDependencies.cmake) add_subdirectory(src) # add test app -add_subdirectory(app) +if(BUILD_MQT_DEBUGGER_APP) + add_subdirectory(app) +endif() # add test code -if(BUILD_MQT_DEBUG_TESTS) +if(BUILD_MQT_DEBUGGER_TESTS) enable_testing() include(GoogleTest) add_subdirectory(test) @@ -52,5 +55,5 @@ endif() configure_file(${CMAKE_CURRENT_SOURCE_DIR}/cmake/cmake_uninstall.cmake.in ${CMAKE_CURRENT_BINARY_DIR}/cmake_uninstall.cmake IMMEDIATE @ONLY) -add_custom_target(uninstall-debug COMMAND ${CMAKE_COMMAND} -P - ${CMAKE_CURRENT_BINARY_DIR}/cmake_uninstall.cmake) +add_custom_target(uninstall-debugger COMMAND ${CMAKE_COMMAND} -P + ${CMAKE_CURRENT_BINARY_DIR}/cmake_uninstall.cmake) diff --git a/README.md b/README.md index b1802c4..01250ae 100644 --- a/README.md +++ b/README.md @@ -1,10 +1,10 @@ -[![PyPI](https://img.shields.io/pypi/v/mqt.debug?logo=pypi&style=flat-square)](https://pypi.org/project/mqt.debug/) +[![PyPI](https://img.shields.io/pypi/v/mqt.debugger?logo=pypi&style=flat-square)](https://pypi.org/project/mqt.debugger/) ![OS](https://img.shields.io/badge/os-linux%20%7C%20macos%20%7C%20windows-blue?style=flat-square) [![License: MIT](https://img.shields.io/badge/license-MIT-blue.svg?style=flat-square)](https://opensource.org/licenses/MIT) -[![CI](https://img.shields.io/github/actions/workflow/status/cda-tum/mqt-debug/ci.yml?branch=main&style=flat-square&logo=github&label=ci)](https://github.com/cda-tum/mqt-debug/actions/workflows/ci.yml) -[![CD](https://img.shields.io/github/actions/workflow/status/cda-tum/mqt-debug/cd.yml?style=flat-square&logo=github&label=cd)](https://github.com/cda-tum/mqt-debug/actions/workflows/cd.yml) -[![Documentation](https://img.shields.io/readthedocs/debug?logo=readthedocs&style=flat-square)](https://mqt.readthedocs.io/projects/debug) -[![codecov](https://img.shields.io/codecov/c/github/cda-tum/mqt-debug?style=flat-square&logo=codecov)](https://codecov.io/gh/cda-tum/mqt-debug) +[![CI](https://img.shields.io/github/actions/workflow/status/cda-tum/mqt-debugger/ci.yml?branch=main&style=flat-square&logo=github&label=ci)](https://github.com/cda-tum/mqt-debugger/actions/workflows/ci.yml) +[![CD](https://img.shields.io/github/actions/workflow/status/cda-tum/mqt-debugger/cd.yml?style=flat-square&logo=github&label=cd)](https://github.com/cda-tum/mqt-debugger/actions/workflows/cd.yml) +[![Documentation](https://img.shields.io/readthedocs/debugger?logo=readthedocs&style=flat-square)](https://mqt.readthedocs.io/projects/debugger) +[![codecov](https://img.shields.io/codecov/c/github/cda-tum/mqt-debugger?style=flat-square&logo=codecov)](https://codecov.io/gh/cda-tum/mqt-debugger)

@@ -15,54 +15,52 @@

-# MQT Debug - A Quantum Circuit Debugging Tool +# MQT Debugger - A Quantum Circuit Debugging Tool A tool for debugging quantum circuits developed as part of the [_Munich Quantum Toolkit (MQT)_](https://mqt.readthedocs.io) by the [Chair for Design Automation](https://www.cda.cit.tum.de/) at the [Technical University of Munich](https://www.tum.de/). It proposes an interface for the simulation of circuits and diagnosis of errors and provides a base implementation built upon [MQT Core](https://github.com/cda-tum/mqt-core), which forms the backbone of the MQT. It also provides a Debugger Adapter Protocol (DAP) server that can be used to integrate the debugger into IDEs.

- + Documentation

-If you have any questions, feel free to contact us via [quantum.cda@xcit.tum.de](mailto:quantum.cda@xcit.tum.de) or by creating an issue on [GitHub](https://github.com/cda-tum/mqt-debug/issues). +If you have any questions, feel free to contact us via [quantum.cda@xcit.tum.de](mailto:quantum.cda@xcit.tum.de) or by creating an issue on [GitHub](https://github.com/cda-tum/mqt-debugger/issues). ## Getting Started -MQT Debug is available via [PyPI](https://pypi.org/project/mqt.debug/) for Linux, macOS, and Windows and supports Python 3.8 to 3.12. +MQT Debugger is available via [PyPI](https://pypi.org/project/mqt.debugger/) for Linux, macOS, and Windows and supports Python 3.8 to 3.12. ```console -(venv) $ pip install mqt.debug +(venv) $ pip install mqt.debugger ``` The following code gives an example on the usage: ```python3 -from mqt import debug +from mqt import debugger -state = debug.create_ddsim_simulation_state() +state = debugger.create_ddsim_simulation_state() with open("code.qasm", "r") as f: state.load_code(f.read()) f.run_simulation() print(f.get_state_vector_full()) ``` -**Detailed documentation on all available methods, options, and input formats is available at [ReadTheDocs](https://mqt.readthedocs.io/projects/debug).** +**Detailed documentation on all available methods, options, and input formats is available at [ReadTheDocs](https://mqt.readthedocs.io/projects/debugger).** ## System Requirements and Building The implementation is compatible with any C++20 compiler, a minimum CMake version of 3.19, and Python 3.8+. -Please refer to the [documentation](https://mqt.readthedocs.io/projects/debug) on how to build the project. +Please refer to the [documentation](https://mqt.readthedocs.io/projects/debugger) on how to build the project. Building (and running) is continuously tested under Linux, macOS, and Windows using the [latest available system versions for GitHub Actions](https://github.com/actions/virtual-environments). ## References -MQT Debug has been developed based on methods proposed in the following papers: - -TODO +MQT Debugger has been developed based on methods proposed in a scientific paper that has not been released yet. ## Acknowledgements diff --git a/app/CMakeLists.txt b/app/CMakeLists.txt index 62deed2..25ba47c 100644 --- a/app/CMakeLists.txt +++ b/app/CMakeLists.txt @@ -1,10 +1,10 @@ -add_executable(mqt_debug_app testDDSimDebugger.cpp) +add_executable(mqt_debugger_app testDDSimDebugger.cpp) # set include directories -target_include_directories(mqt_debug_app PUBLIC ${PROJECT_SOURCE_DIR}/include - ${PROJECT_BINARY_DIR}/include) +target_include_directories(mqt_debugger_app PUBLIC ${PROJECT_SOURCE_DIR}/include + ${PROJECT_BINARY_DIR}/include) # link to the MQT::Core libraries -target_link_libraries(mqt_debug_app PRIVATE ${PROJECT_NAME}) -target_link_libraries(mqt_debug_app PUBLIC MQT::CoreDD) -target_link_libraries(mqt_debug_app PRIVATE MQT::ProjectWarnings MQT::ProjectOptions) +target_link_libraries(mqt_debugger_app PRIVATE ${PROJECT_NAME}) +target_link_libraries(mqt_debugger_app PUBLIC MQT::CoreDD) +target_link_libraries(mqt_debugger_app PRIVATE MQT::ProjectWarnings MQT::ProjectOptions) diff --git a/app/testDDSimDebugger.cpp b/app/testDDSimDebugger.cpp index 7c2ea03..b3d31be 100644 --- a/app/testDDSimDebugger.cpp +++ b/app/testDDSimDebugger.cpp @@ -8,19 +8,25 @@ #include #include -#include +#include #include int main() { - std::ifstream file("../../app/code/test" - ".qasm"); + std::ifstream file("program.qasm"); + if (!file.is_open()) { + file.open("../../app/code/test" + ".qasm"); + } if (!file.is_open()) { std::cerr << "Could not open file\n"; file.close(); return 1; } - const std::string code((std::istreambuf_iterator(file)), - std::istreambuf_iterator()); + + std::stringstream buffer; + buffer << file.rdbuf(); + + const std::string code = buffer.str(); DDSimulationState state; createDDSimulationState(&state); diff --git a/cmake/ExternalDependencies.cmake b/cmake/ExternalDependencies.cmake index a15a016..1d0fcfc 100644 --- a/cmake/ExternalDependencies.cmake +++ b/cmake/ExternalDependencies.cmake @@ -1,7 +1,7 @@ include(FetchContent) set(FETCH_PACKAGES "") -if(BUILD_MQT_DEBUG_BINDINGS) +if(BUILD_MQT_DEBUGGER_BINDINGS) if(NOT SKBUILD) # Manually detect the installed pybind11 package. execute_process( @@ -87,7 +87,7 @@ else() endif() endif() -if(BUILD_MQT_DEBUG_TESTS) +if(BUILD_MQT_DEBUGGER_TESTS) set(gtest_force_shared_crt ON CACHE BOOL "" FORCE) @@ -107,7 +107,7 @@ if(BUILD_MQT_DEBUG_TESTS) endif() endif() -if(BUILD_MQT_DEBUG_BINDINGS) +if(BUILD_MQT_DEBUGGER_BINDINGS) # add pybind11_json library if(CMAKE_VERSION VERSION_GREATER_EQUAL 3.24) FetchContent_Declare( diff --git a/docs/.gitignore b/docs/.gitignore new file mode 100644 index 0000000..cf076a7 --- /dev/null +++ b/docs/.gitignore @@ -0,0 +1,2 @@ +build/ +doxygen/ diff --git a/docs/Doxyfile b/docs/Doxyfile new file mode 100644 index 0000000..6e7c9e7 --- /dev/null +++ b/docs/Doxyfile @@ -0,0 +1,143 @@ +# Doxyfile 1.9.8 + +# This file describes the settings to be used by the documentation system +# doxygen (www.doxygen.org) for a project. +# +# All text after a double hash (##) is considered a comment and is placed in +# front of the TAG it is preceding. +# +# All text after a single hash (#) is considered a comment and will be ignored. +# The format is: +# TAG = value [value, ...] +# For lists, items can also be appended using: +# TAG += value [value, ...] +# Values that contain spaces should be placed between quotes (\" \"). + +#--------------------------------------------------------------------------- +# Project related configuration options +#--------------------------------------------------------------------------- + +# The PROJECT_NAME tag is a single word (or a sequence of words surrounded by +# double-quotes, unless you are using Doxywizard) that should identify the +# project for which the documentation is generated. This name is used in the +# title of most generated pages and in a few other places. +# The default value is: My Project. + +PROJECT_NAME = "MQT Debugger" + +# Using the PROJECT_BRIEF tag one can provide an optional one line description +# for a project that appears at the top of each page and should give viewer a +# quick idea about the purpose of the project. Keep the description short. + +PROJECT_BRIEF = "A quantum circuit simulator based on decision diagrams written in C++" + +# The OUTPUT_DIRECTORY tag is used to specify the (relative or absolute) path +# into which the generated documentation will be written. If a relative path is +# entered, it will be relative to the location where doxygen was started. If +# left blank the current directory will be used. + +OUTPUT_DIRECTORY = doxygen + +#--------------------------------------------------------------------------- +# Configuration options related to the input files +#--------------------------------------------------------------------------- + +# The INPUT tag is used to specify the files and/or directories that contain +# documented source files. You may enter file names like myfile.cpp or +# directories like /usr/src/myproject. Separate the files or directories with +# spaces. See also FILE_PATTERNS and EXTENSION_MAPPING +# Note: If this tag is empty the current directory is searched. + +INPUT = ../include ../src + +# If the value of the INPUT tag contains directories, you can use the +# FILE_PATTERNS tag to specify one or more wildcard patterns (like *.cpp and +# *.h) to filter out the source-files in the directories. +# +# Note that for custom extensions or not directly supported extensions you also +# need to set EXTENSION_MAPPING for the extension otherwise the files are not +# read by doxygen. +# +# Note the list of default checked file patterns might differ from the list of +# default file extension mappings. +# +# If left blank the following patterns are tested:*.c, *.cc, *.cxx, *.cxxm, +# *.cpp, *.cppm, *.c++, *.c++m, *.java, *.ii, *.ixx, *.ipp, *.i++, *.inl, *.idl, +# *.ddl, *.odl, *.h, *.hh, *.hxx, *.hpp, *.h++, *.ixx, *.l, *.cs, *.d, *.php, +# *.php4, *.php5, *.phtml, *.inc, *.m, *.markdown, *.md, *.mm, *.dox (to be +# provided as doxygen C comment), *.py, *.pyw, *.f90, *.f95, *.f03, *.f08, +# *.f18, *.f, *.for, *.vhd, *.vhdl, *.ucf, *.qsf and *.ice. + +FILE_PATTERNS = *.c \ + *.cc \ + *.cxx \ + *.cpp \ + *.c++ \ + *.hh \ + *.hxx \ + *.hpp \ + *.h++ \ + *.h \ + *.py \ + *.pyi + +# The RECURSIVE tag can be used to specify whether or not subdirectories should +# be searched for input files as well. +# The default value is: NO. + +RECURSIVE = YES + +#--------------------------------------------------------------------------- +# Configuration options related to the HTML output +#--------------------------------------------------------------------------- + +# If the GENERATE_HTML tag is set to YES, doxygen will generate HTML output +# The default value is: YES. + +GENERATE_HTML = NO + +#--------------------------------------------------------------------------- +# Configuration options related to the XML output +#--------------------------------------------------------------------------- + +# If the GENERATE_XML tag is set to YES, doxygen will generate an XML file that +# captures the structure of the code including all documentation. +# The default value is: NO. + +GENERATE_XML = YES + +# The XML_OUTPUT tag is used to specify where the XML pages will be put. If a +# relative path is entered the value of OUTPUT_DIRECTORY will be put in front of +# it. +# The default directory is: xml. +# This tag requires that the tag GENERATE_XML is set to YES. + +XML_OUTPUT = xml + +#--------------------------------------------------------------------------- +# Configuration options related to diagram generator tools +#--------------------------------------------------------------------------- + +# If set to YES the inheritance and collaboration graphs will hide inheritance +# and usage relations if the target is undocumented or is not a class. +# The default value is: YES. + +HIDE_UNDOC_RELATIONS = YES + +# If you set the HAVE_DOT tag to YES then doxygen will assume the dot tool is +# available from the path. This tool is part of Graphviz (see: +# https://www.graphviz.org/), a graph visualization toolkit from AT&T and Lucent +# Bell Labs. The other options in this section have no effect if this option is +# set to NO +# The default value is: YES. + +HAVE_DOT = NO + +#--------------------------------------------------------------------------- +# Configuration options related to the LaTeX output +#--------------------------------------------------------------------------- + +# If the GENERATE_LATEX tag is set to YES, doxygen will generate LaTeX output. +# The default value is: YES. + +GENERATE_LATEX = NO diff --git a/docs/source/Assertions.rst b/docs/source/Assertions.rst new file mode 100644 index 0000000..8f5e3e0 --- /dev/null +++ b/docs/source/Assertions.rst @@ -0,0 +1,116 @@ +Assertions +========== + +This document describes the syntax and semantics of assertions in MQT Debugger. +Assertions are a useful tool to test the correctness of programs by comparing the current state of the system to an expected state. +The assertions in MQT Debugger allow developers to test either the exact or approximate state or certain properties, such as entanglement or superposition. + +The following sections will give an overview of the assertion syntax and the different types of assertions that can be used in MQT Debugger. + +.. note:: + + The targets of an assertion can be either individual qubits or full quantum registers. However, when the selected targets are subset of the full quantum state, + then none of the qubits in this sub-state is allowed to be entangled with any other qubits outside of the sub-state. This is, because in many cases, exact sub-states + as state vectors are not well-defined if they are entangled with other qubits. + +Entanglement Assertion +###################### + +Entanglement assertions check whether a set of qubits is entangled. For this purpose, every qubit in the set is compared to all other qubits in the set, and +entanglement must exist between each possible pair. + +**QASM Syntax**: + +.. code-block:: + + assert-ent [, ...]*; + +*Example*: + +.. code-block:: + + assert-ent q[0], q[1], q[2]; + +This assertions checks for entanglement between qubits ``q[0]`` and ``q[1]``, ``q[0]`` and ``q[2]``, as well as ``q[1]`` and ``q[2]``. + +An example for a quantum state that would pass this assertion is the GHZ state: :math:`\frac{1}{\sqrt{2}}(|000\rangle + |111\rangle)`. +As none of the individual qubits is separable, this state is entangled. + +An example for a quantum state that would fail this assertion is the state :math:`\frac{1}{\sqrt{2}}(|000\rangle + |110\rangle)`. +As the least-significant qubit is separable, this state is not entangled. + + +Superposition Assertion +####################### + +Superposition assertions check whether a set of qubits is in a superposition state. +A set of qubits is in a superposition state if there exist at least two basis states of the full system that have different assignments for the target qubits and a non-zero amplitude. +This means, that not every qubit in the set must be in a superposition state individually, but the full set must be in a superposition state. + +**QASM Syntax**: + +.. code-block:: + + assert-sup [, ...]*; + +*Example*: + +.. code-block:: + + assert-sup q[0], q[1], q[2]; + +This assertion checks for superposition of qubits ``q[0]``, ``q[1]``, and ``q[2]``. + +An example for a quantum state that would pass this assertion is the state :math:`\frac{1}{\sqrt{2}}(|0\rangle + |1\rangle) \otimes |0\rangle`. +As two basis states have non-zero amplitudes (:math:`|00\rangle` and :math:`|10\rangle`), this state is in a superposition. + +An example for a quantum state that would fail this assertion is the state :math:`|00\rangle`. +In this case, only a single state (:math:`|00\rangle`) has a non-zero amplitude, so the state is not in a superposition. + + +Equality Assertion +################## + +Equality assertions compare the state of a set of qubits to a given state and fail if the states are not equal. +Furthermore, a similarity threshold can be passed to the assertion, allowing for approximate comparisons. The similarity is computed through the +cosine similarity of two functions and can be set between 0 and 1. If no similarity threshold is passed, the default value of 1 is used. + +The target state to compare to can be expressed as a state vector or as a new quantum circuit. +**QASM Syntax**: + +.. code-block:: + + assert-eq [similarity], [, ...]* { STATE_REPRESENTATION } + +.. code-block:: + + STATE_REPRESENTATION = + | + | + + STATEVECTOR = [, ...]* + + CIRCUIT = + +If the selected state representation is a state vector, it must represent a system of the same number of qubits as the assertion compares to +and use :math:`2^n` amplitudes. Amplitudes can be real or complex numbers using :math:`i` or :math:`j` for the imaginary unit. + +If the selected state representation is a circuit, it must be defined as a new quantum program in QASM format. The circuit must not contain any further assertions and +must use the same number of qubits as the assertion compares to. The current system state will then be compared to the state after running the circuit. + +*Example*: + +.. code-block:: + + assert-eq 0.9, q[0], q[1] { 0.5, 0.5, 0.5, 0.5 }; + +This assertion checks whether the state of qubits ``q[0]`` and ``q[1]`` is equal to the state :math:`\frac{1}{2}(|00\rangle + |01\rangle + |10\rangle + |11\rangle)` with a similarity threshold of 0.9. + +.. code-block:: + + assert-eq q[0], q[1] { + h q[0]; + cx q[0], q[1]; + }; + +This assertion checks whether the state of qubits ``q[0]`` and ``q[1]`` is equal to the bell state :math:`\frac{1}{\sqrt{2}}(|00\rangle + |11\rangle)`. diff --git a/docs/source/Contributing.rst b/docs/source/Contributing.rst new file mode 100644 index 0000000..1da5d54 --- /dev/null +++ b/docs/source/Contributing.rst @@ -0,0 +1 @@ +.. include:: ../../.github/contributing.rst diff --git a/docs/source/Debugging.rst b/docs/source/Debugging.rst new file mode 100644 index 0000000..dd040da --- /dev/null +++ b/docs/source/Debugging.rst @@ -0,0 +1,88 @@ +Interactive Debugging +===================== + +MQT Debugger provides various methods to interactively debug quantum programs. +It can be used to step through the program and inspect the state of the system. +This document further details the debugging capabilities provided by this framework. + +As MQT Debugger currently only supports OpenQASM as input language, this document will give examples and details related to OpenQASM programs. +However, due to the modular nature of this framework, it can be extended to support other languages as well. + +.. _stepping: + +Stepping Through the Program +---------------------------- + +The most basic debugging feature is stepping through the program. +Like many typical interactive debuggers, this framework supports three types of steps: + +- **Single Step**: Executes the next instruction and stops afterwards. If the next instruction is a function call, it will step into the function. +- **Step Over**: Executes the next instruction and stops afterwards. If the next instruction is a function call, it will execute the entire function at once. +- **Step Out**: Continues execution until the current function returns. + +However, due to the reversibility of quantum programs, the debugger can also step backward: + +- **Single Step Backward**: Reverts the last instruction and stops afterwards. If the last instruction was a function call, it will only stop at the last instruction of that function. +- **Step Over Backward**: Reverts the last instruction and stops afterwards. If the last instruction was a function call, it will revert the entire function at once. +- **Step Out Backward**: Reverts all instructions in the current function scope and stops at the instruction calling the function of the current scope. + +Continuing and Pausing Execution +-------------------------------- + +In contrast to individual steps, the debugger can also be used to continue the entire program execution, using +the :cpp:member:`SimulationState::runSimulation `/:py:meth:`SimulationState.run_simulation ` method. +This will execute the program until the end or until a :ref:`breakpoint ` or a :ref:`failing assertion ` is reached. + +Additionally, the :cpp:member:`SimulationState::runAll `/:py:meth:`SimulationState.run_all ` method can be used to run the program without stopping, +instead counting how many failing assertions were encountered. + +Furthermore, the :cpp:member:`SimulationState::pauseSimulation `/:py:meth:`SimulationState.pause_simulation ` method can be used to pause the execution at any point in time. + +Inspecting the State +-------------------- + +MQT Debugger distinguishes between classical variables and quantum variables. For OpenQASM, currently only boolean classical variables are supported. + +The framework provides different methods to inspect the state of the system at runtime. Classical variables can accessed using the +:cpp:member:`SimulationState::getClassicalVariable `/:py:meth:`SimulationState.get_classical_variable ` method, passing the name of the desired variable, +which returns an object representing the variable. + +Quantum variables cannot be accessed directly, but the developer can instead access the statevector to inspect the quantum state at any point in time. +:cpp:member:`SimulationState::getStateVectorFull `/:py:meth:`SimulationState.get_state_vector_full ` can be used to obtain the full statevector of the system. As this statevector can +be very large, :cpp:member:`SimulationState::getStateVectorSub `/:py:meth:`SimulationState.get_state_vector_sub ` can be used to obtain a sub-statevector of the system, containing +just a subset of all qubits. In this case, the qubits included in the sub-statevector must not be entangled with any qubits outside it. + +Furthermore, the framework also allows to inspect individual amplitude values of the statevector using +:cpp:member:`SimulationState::getAmplitudeIndex `:py:meth:`SimulationState.get_amplitude_index ` or :cpp:member:`SimulationState::getAmplitudeBitstring `/:py:meth:`SimulationState.get_amplitude_bitstring `. +In these cases, the developer must identify the desired amplitude by passing either the index of the amplitude or the bitstring that represents the desired state. + +.. _breakpoints: + +Breakpoints +----------- + +Breakpoints can be set to force execution to stop at a specific instruction. To be compatible with the typical protocols, setting a breakpoint +requires the character index in the source code, at which the breakpoint should be set. MQT Debugger will then determine the instruction that +corresponds to this location in the code and stop execution there in the future. + +To set a breakpoint, the :cpp:member:`SimulationState::setBreakpoint `/:py:meth:`SimulationState.set_breakpoint ` method can be used, passing the desired character index. This +method will return the instruction index at which the breakpoint was set (Python) or store it in the provided reference (C++). + +To remove breakpoints, the :cpp:member:`SimulationState::clearBreakpoints `/:py:meth:`SimulationState.clear_breakpoints ` method can be used, removing all breakpoints. + +When a program is paused during execution, the methods :cpp:member:`SimulationState::wasBreakpointHit `/:py:meth:`SimulationState.was_breakpoint_hit ` can be used to check whether the +current pause was caused by a breakpoint. This allows developers to distinguish between pauses due to breakpoints and pauses due to other reasons, such as :ref:`failed assertions `. + +.. _assertions: + +Assertions +---------- + +MQT Debugger supports :doc:`assertions ` to check the state of the system at runtime. +If an assertion fails, the debugger will pause execution **before** the failing instruction. + +Any methods to :ref:`step through the program ` will pause as a failing assertion. Continuing execution after hitting a failed assertion will skip +the failing instruction and continue with the next one. + +When a program is paused during execution, the methods :cpp:member:`SimulationState::didAssertionFail `/:py:meth:`SimulationState.did_assertion_fail ` can be used to check whether the +current pause was caused by a failing assertion. This allows developers to distinguish between pauses due to assertions and pauses due to other reasons, such as :ref:`breakpoints `. diff --git a/docs/source/DevelopmentGuide.rst b/docs/source/DevelopmentGuide.rst new file mode 100644 index 0000000..02fa627 --- /dev/null +++ b/docs/source/DevelopmentGuide.rst @@ -0,0 +1,59 @@ +Development Guide +================= + + +Installation +############ + +In order to start developing, clone the MQT Debugger repository using + + .. code-block:: console + + $ git clone https://github.com/cda-tum/mqt-debugger + +A C++ compiler supporting C++17 and a minimum CMake version of 3.19 are required to build the project. + +Working on the core C++ library +############################### + +Our projects use CMake as the main build configuration tool. +Building a project using CMake is a two-stage process. +First, CMake needs to be configured by calling + + .. code-block:: console + + $ cmake -S . -B build -DCMAKE_BUILD_TYPE=Release + +This tells CMake to search the current directory :code:`.` (passed via :code:`-S` for source) for a :code:`CMakeLists.txt` file and process it into a directory :code:`build` (passed via :code:`-B`). +The flag :code:`-DCMAKE_BUILD_TYPE=Release` tells CMake to configure a *Release* build (as opposed to, e.g., a *Debug* build). We also recommend setting the flags +:code:`-DEIGEN_BUILD_TESTING=OFF` and :code:`-DEIGEN_BUILD_DOC=OFF` to prevent the documentation and tests of the ``Eigen3`` library included in the project from being built. + +After configuring with CMake, the project can be built by calling + + .. code-block:: console + + $ cmake --build build --config Release + +This tries to build the project in the :code:`build` directory (passed via :code:`--build`). +Some operating systems and developer environments explicitly require a configuration to be set, which is why the :code:`--config` flag is also passed to the build command. The flag :code:`--parallel ` may be added to trigger a parallel build. + +Building the project this way generates + +- the main library :code:`libmqt_debugger.a` (Unix) / :code:`mqt_debugger.lib` (Windows) in the :code:`build/src` directory +- a test executable :code:`mqt_debugger_test` containing unit tests in the :code:`build/test` directory (this requires passing :code:`-DBUILD_MQT_DEBUGGER_TESTS=ON` to CMake during configuration) +- the Python bindings library :code:`pydebugger.<...>` in the :code:`build/src/python` directory (this requires passing :code:`-DBUILD_MQT_DEBUGGER_BINDINGS=ON` to CMake during configuration) +- the CLI App :code:`mqt_debugger_app` in the :code:`build/app` directory (this requires passing :code:`-DBUILD_MQT_DEBUGGER_APP=ON` to CMake during configuration) + +Working on the Python module +############################ + +The :code:`mqt.debugger` Python module can be conveniently built locally by calling + + .. code-block:: console + + (venv) $ pip install --editable . + +The :code:`--editable` flag ensures that changes in the Python code are instantly available without re-running the command. + +`Pybind11 `_ is used for providing bindings of the C++ core library to Python (see `bindings.cpp `_). +If parts of the C++ code have been changed, the above command has to be run again to make the changes visible in Python. diff --git a/docs/source/Diagnosis.rst b/docs/source/Diagnosis.rst new file mode 100644 index 0000000..3fd09f8 --- /dev/null +++ b/docs/source/Diagnosis.rst @@ -0,0 +1,86 @@ +Diagnosis Methods +================= + +This document describes the diagnostics methods available in MQT Debugger. +The diagnostics methods can be used to analyze the state of the system, in particular to find potential error causes when an :doc:`assertion ` fails. + +All of these methods require a :py:class:`Diagnostics ` object to be executed, which can be obtained from the :py:class:`SimulationState ` object using +:cpp:member:`SimulationState::getDiagnostics `/:py:meth:`SimulationState.get_diagnostics `. + +Error Cause Analysis +##################### + +When an assertion fails, several methods can be used to find potential error clauses. + +.. For further information, please refer to :cite:labelpar:`rovara2025debugging`. + +Cone of Influence Analysis +-------------------------- + +Cone of Influence Analysis partitions a given quantum program into two parts: the part that influences the failed assertion and the part that does not. +This analysis method can be used to reduce the number of instructions that need to be checked for potential error causes. If the Cone of Influence analysis manages to +narrow down just a small subset of the program, the error cause can be found more quickly. Furthermore, other analysis methods use the Cone of Influence as input +to find potential error causes more efficiently. + +The cone of influence can be obtained using :cpp:member:`Diagnostics::getDataDependencies `/:py:meth:`Diagnostics.get_data_dependencies `. +In the Python version, it requires a single instruction to be passed as an argument, and returns the indices of all instructions that influence it. +In the C++ version, in addition to the desired instruction, a pointer to a boolean array must be passed. Each element of the array corresponds to an instruction in the program and is set to ``true`` if the instruction is part of the cone of influence. +Finally, a boolean Flag (``includeCallers``) can be passed to also include instructions outside the current scope in the cone of influence. Otherwise, if the function is called +inside a custom gate definition, it will only include data dependencies within the gate. + +The cone of influence is computed by recursively iterating over the instructions in the current cone of influence and adding all instructions that influence them, until it reaches a fixed point. + +Interaction Analysis +-------------------- + +Interaction Analysis is a method to find potential error causes by analyzing the interactions between qubits in the system. +It is automatically called when using :cpp:member:`Diagnostics::potentialErrorCauses `/:py:meth:`Diagnostics.potential_error_causes `. + +This analysis method can be used to find reasons for failing entanglement assertions: +Whenever a failed entanglement assertion is encountered, the Interaction Analysis checks, whether the target qubits of the assertion interact with each other. + +If this is not the case, then clearly, no entanglement can be prepared, so it is treated as a potential error cause. The following code shows an example situation, +in which Interaction Analysis would find a potential error cause: + +.. code-block:: + + qreg q[3]; + + h q[0]; + cx q[0], q[1]; + + assert-ent q; + +Here, calling ``potential_error_causes()`` yields two errors: ``Missing interaction between q[0] and q[2]`` and ``Missing interaction between q[1] and q[2]``. + +.. note:: + Interaction Analysis is generally a static analysis method. However, when it is performed at runtime for the `potential_error_causes()` method, + it further uses dynamically obtained information to improve its results. During execution, the diagnostics tool keeps track of all actual qubits that + were involved in instructions, even inside custom gate definitions, where static analysis would not be able to determine the exact qubits involved. + This way, it can extend the interaction analysis throughout the entire program, even in other scopes. This is not always possible when performing interaction analysis statically. + +Control-Value Analysis +---------------------- + +Control-Value Analysis is a method that dynamically analyzes the program during execution to find incorrectly defined controlled gates. +In particular, it looks for controlled gates for which the control is always purely in the state :math:`|0\rangle`. In these cases, +the controlled gate will never affect the full state, which could be a sign for an error. + +This analysis also similarly checks for inverse-controlled gates (i.e., controlled gates that tirgger when the control value is :math:`|1\rangle`) that always +have the state :math:`|0\rangle` as control. + +It is automatically called when using :cpp:member:`Diagnostics::potentialErrorCauses `/:py:meth:`Diagnostics.potential_error_causes `. + +The following code shows an example situation, in which Control-Value Analysis would find a potential error cause: + +.. code-block:: + + qreg q[3]; + + h q[0]; + cx q[0], q[1]; + cx q[2], q[0]; + + assert-ent q; + +Here, calling ``potential_error_causes()`` yields the error ``Controlled gate with constant control value`` for instruction ``cx q[2], q[0]``. diff --git a/docs/source/Installation.rst b/docs/source/Installation.rst new file mode 100644 index 0000000..5e2649d --- /dev/null +++ b/docs/source/Installation.rst @@ -0,0 +1,96 @@ +Installation +============ + +MQT Debugger is mainly developed as a C++ library with many of its interfaces for important modules being defined in C. +Parts of the implementation build upon `MQT Core `_, which forms the backbone of the `MQT `_. +In order to make the tool as accessible as possible, it comes with an easy-to-use Python interface. + +We encourage installing MQT Debugger via pip (preferably in a `virtual environment `_): + + .. code-block:: console + + (venv) & pip install mqt.debugger + +In most practical cases (under 64-bit Linux, MacOS incl. Apple Silicon, and Windows), this requires no compilation and merely downloads and installs a platform-specific pre-built wheel. + +.. note:: + In order to set up a virtual environment, you can use the following commands: + + .. code-block:: console + + $ python3 -m venv venv + $ source venv/bin/activate + + If you are using Windows, you can use the following commands instead: + + .. code-block:: console + + $ python3 -m venv venv + $ venv\Scripts\activate.bat + + It is recommended to make sure that you are using the latest version of pip, setuptools, and wheel before trying to install the project: + + .. code-block:: console + + (venv) $ pip install --upgrade pip setuptools wheel + +A Detailed Walk Through +####################### + +First, save the following lines as :code:`ghz.qasm` in a folder where you want to install MQT Debugger: + + .. code-block:: + + qreg q[3]; + + h q[0]; + cx q[0], q[1]; + cx q[2], q[0]; + + assert-ent q; + +Then, create the following Python script as :code:`debug_ghz.py` in the same folder: + + .. code-block:: python + + import mqt.debugger as dbg + + state = dbg.create_ddsim_simulation_state() + with open("ghz.qasm") as f: + state.load_code(f.read()) + + state.run_simulation() + + if state.did_assertion_fail(): + print(f"Assertion failed at instruction {state.get_current_instruction() + 1}") + problems = state.get_diagnostics().potential_error_causes() + print("Potential Errors:") + for problem in problems: + print(f"{problem.type} at instruction {problem.instruction + 1}") + +The following snippet shows the installation process from setting up the virtual environment to running a small example program. + + .. code-block:: console + + $ python3 -m venv venv + $ . venv/bin/activate + (venv) $ pip install -U pip setuptools wheel + (venv) $ pip install mqt.debugger + (venv) $ python3 debug_ghz.py + +Building from Source for Performance +#################################### + +In order to get the best performance out of MQT Debugger and enable platform-specific compiler optimizations that cannot be enabled on portable wheels, it is recommended to build the package from source via: + + .. code-block:: console + + (venv) $ pip install mqt.debugger --no-binary mqt.debugger + +This requires a `C++ compiler `_ compiler supporting *C++17* and a minimum `CMake `_ version of *3.19*. + +The library is continuously tested under Linux, MacOS, and Windows using the `latest available system versions for GitHub Actions `_. +In order to access the latest build logs, visit `mqt-debugger/actions/workflows/ci.yml `_. + +.. note:: + We noticed some issues when compiling with Microsoft's *MSCV* compiler toolchain. If you want to start development on this project under Windows, consider using the *clang* compiler toolchain. A detailed description of how to set this up can be found `here `_. diff --git a/docs/source/Publications.rst b/docs/source/Publications.rst new file mode 100644 index 0000000..84ee863 --- /dev/null +++ b/docs/source/Publications.rst @@ -0,0 +1,14 @@ +Publications +============ + +.. *MQT Debugger* is academic software. Thus, many of its built-in algorithms have been published as scientific papers. + +.. *MQT Debugger* is academic software. Thus, its built-in algorithms have been published as a scientific paper. + +*MQT Debugger* is academic software. Thus, many of its built-in algorithms are being published as scientific papers. Currently, these publications are still under review but will be listed here in the future. + +.. If you use *MQT Debugger* in your work, we would appreciate if you cited: + +.. If you use *MQT Debugger* in your work, we would appreciate if you cited: :cite:labelpar:`rovara2025debugging`. + +.. bibliography:: diff --git a/docs/source/Quickstart.rst b/docs/source/Quickstart.rst new file mode 100644 index 0000000..de73699 --- /dev/null +++ b/docs/source/Quickstart.rst @@ -0,0 +1,100 @@ +Quickstart +========== + +This documentation gives a quick overview on how to get started with MQT Debugger using: +- The Python library +- The DAP server +- The CLI app + +Python Library +############## + +The Python bindings are offered to give an easy start into using MQT Debugger. + +Working with the Python library of MQT Debugger requires several preparation steps: + +.. code-block:: python + + import mqt.debugger as dbg + +All main functionalities are included in the module ``mqt.debugger``, no other modules need to be imported. + +.. code-block:: python + + state = dbg.create_ddsim_simulation_state() + +The first step is to create a simulation state. The current implementation of MQT Debugger implements a single simulation backend based on +decision diagrams from `MQT Core `_. This backend is instantiated by calling :py:func:`create_ddsim_simulation_state `. + +.. code-block:: python + + state.load_code(your_code) + +Before running the debugger, a quantum program must be loaded into the state. This is done by calling :py:meth:`SimulationState.load_code ` with the quantum program as a string argument. +Currently, the supported quantum program format is `QASM 2.0 `_. + +After this setup is done, the debugging process can be started by stepping through the code one instruction at a time: + +.. code-block:: python + + state.step_forward() + +or by running the full simulation: + +.. code-block:: python + + state.run_simulation() + +In the second case, simulation will be run until the program ends, a failing assertion is encountered, or a breakpoint is reached. +Further details on how to step through the program can be found in the :doc:`reference documentation `. + +Breakpoints can be set using + +.. code-block:: python + + state.set_breakpoint(character_index) + +where ``character_index`` is the index of the character in the original code's string, at which the breakpoint should be set. + +Assertions can be added to the code following the :doc:`assertion syntax ` and are automatically evaluated. + +When an assertion fails, the diagnostics methods can be used to get more information about the failure. To this end, first access the diagnostics interface: + +.. code-block:: python + + diagnostics = state.get_diagnostics() + +Then, the potential error causes can be retrieved: + +.. code-block:: python + + problems = diagnostics.potential_error_causes() + print(problems) + +DAP Server +########## + +This library provides a DAP Server that can be connected to from existing IDEs like Visual Studio Code or CLion. + +It can be started by calling + +.. code-block:: console + + python3 -m mqt.debugger.dap.adapter + +The server will then start on port 4711 and can accept one single connection from debugging clients. + +.. note:: + Connecting to the server requires a client compatible with the `Debug Adapter Protocol `_. + While most common IDEs already support it by default, some additional setup or extensions may be required to allow communication with arbitrary clients. + +The DAP Server provides all simulation methods that are accessible via the Python library. +On assertion failures, the server will automatically pause the simulation and send a message to the client containing possible error causes. + +CLI App +####### + +The CLI app is a standalone application that can be used to debug quantum programs from the command line. It is mainly supposed to be used as a testing tool +and thus does not provide all the features of the framework or full accessibility through CLI parameters. + +Instead, the CLI app will open a OpenQASM file with the name :code:`program.qasm` in the current working directory and start the debugging process for it. diff --git a/docs/source/Support.rst b/docs/source/Support.rst new file mode 100644 index 0000000..a88fd1a --- /dev/null +++ b/docs/source/Support.rst @@ -0,0 +1 @@ +.. include:: ../../.github/support.rst diff --git a/docs/source/_static/custom.css b/docs/source/_static/custom.css new file mode 100644 index 0000000..4aeb0aa --- /dev/null +++ b/docs/source/_static/custom.css @@ -0,0 +1,40 @@ +.acknowledgements { + margin-top: 1rem; + padding-bottom: 1rem; + padding-top: 1rem; + border-top: 1px solid var(--color-background-border); + font-size: var(--font-size--small); + color: var(--color-foreground-secondary); +} + +.acknowledgements-logos { + display: grid; + grid-template-columns: repeat(auto-fit, minmax(100px, 1fr)); + grid-gap: 1em; + align-items: center; + margin-top: 0.5rem; +} +.acknowledgement { + display: flex; + flex-direction: column; + align-items: center; + justify-content: center; +} + +/* override the default background color for literal strings */ +body:not([data-theme="light"]) .highlight .sa, +.highlight .sb, +.highlight .sc, +.highlight .dl, +.highlight .sd, +.highlight .s2, +.highlight .se, +.highlight .sh, +.highlight .si, +.highlight .sx, +.highlight .sr, +.highlight .s1, +.highlight .ss, +.highlight .s1 { + background-color: #00000001; +} diff --git a/docs/source/_static/mqt_dark.png b/docs/source/_static/mqt_dark.png new file mode 100644 index 0000000000000000000000000000000000000000..adc910ce84afc340b1cf899d40528786d6d971f7 GIT binary patch literal 61137 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 literal 0 HcmV?d00001 diff --git a/docs/source/_static/mqt_light.png b/docs/source/_static/mqt_light.png new file mode 100644 index 0000000000000000000000000000000000000000..be279e65bd931c10b154c39e76535a9312fdda78 GIT binary patch literal 57266 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 literal 0 HcmV?d00001 diff --git a/docs/source/_templates/page.html b/docs/source/_templates/page.html new file mode 100644 index 0000000..4aad54d --- /dev/null +++ b/docs/source/_templates/page.html @@ -0,0 +1,76 @@ +{% extends "furo/page.html" %} {% block footer %} + +
+ The Munich Quantum Toolkit has been supported by the European Research Council + (ERC) under the European Union's Horizon 2020 research and innovation program + (grant agreement No. 101001318), the Bavarian State Ministry for Science and + Arts through the Distinguished Professorship Program, as well as the Munich + Quantum Valley, which is supported by the Bavarian state government with funds + from the Hightech Agenda Bayern Plus. +
+
+ TUM Logo +
+
+ TUM Logo +
+
+ Coat of Arms of Bavaria +
+
+ ERC Logo +
+
+ ERC Logo +
+
+ MQV Logo +
+
+
+{% endblock footer %} diff --git a/docs/source/conf.py b/docs/source/conf.py new file mode 100644 index 0000000..c51e87b --- /dev/null +++ b/docs/source/conf.py @@ -0,0 +1,153 @@ +"""Sphinx configuration file.""" + +from __future__ import annotations + +import warnings +from importlib import metadata +from pathlib import Path +from typing import TYPE_CHECKING + +import pybtex.plugin +from pybtex.style.formatting.unsrt import Style as UnsrtStyle +from pybtex.style.template import field, href + +if TYPE_CHECKING: + from pybtex.database import Entry + from pybtex.richtext import HRef + +ROOT = Path(__file__).parent.parent.resolve() + +set_type_checking_flag = True + + +try: + from mqt.debugger import __version__ as version +except ModuleNotFoundError: + try: + version = metadata.version("mqt.debugger") + except ModuleNotFoundError: + msg = ( + "Package should be installed to produce documentation! " + "Assuming a modern git archive was used for version discovery." + ) + warnings.warn(msg, stacklevel=1) + + from setuptools_scm import get_version + + version = get_version(root=str(ROOT), fallback_root=ROOT) + +# Filter git details from version +release = version.split("+")[0] + +project = "MQT Debugger" +author = "Damian Rovara" +language = "en" +project_copyright = "Chair for Design Automation, Technical University of Munich" + +master_doc = "index" + +templates_path = ["_templates"] +html_css_files = ["custom.css"] + +extensions = [ + "sphinx.ext.napoleon", + "sphinx.ext.autodoc", + "sphinx.ext.autosummary", + "sphinx.ext.mathjax", + "sphinx.ext.intersphinx", + "sphinx.ext.autosectionlabel", + "sphinx.ext.viewcode", + "sphinx.ext.githubpages", + "sphinxcontrib.bibtex", + "sphinx_copybutton", + "hoverxref.extension", + "nbsphinx", + "sphinxext.opengraph", + "sphinx_autodoc_typehints", + "breathe", +] + +breathe_projects = {"mqt-debugger": str(ROOT / "doxygen/xml")} +breathe_default_project = "MQT Debugger" + +pygments_style = "colorful" + +add_module_names = False + +modindex_common_prefix = ["mqt.debugger."] + +intersphinx_mapping = { + "python": ("https://docs.python.org/3", None), + "typing_extensions": ("https://typing-extensions.readthedocs.io/en/latest", None), + "qiskit": ("https://qiskit.org/documentation", None), + "mqt": ("https://mqt.readthedocs.io/en/latest", None), + "core": ("https://mqt.readthedocs.io/projects/core/en/latest", None), + "ddsim": ("https://mqt.readthedocs.io/projects/ddsim/en/latest", None), + "qmap": ("https://mqt.readthedocs.io/projects/qmap/en/latest", None), + "qecc": ("https://mqt.readthedocs.io/projects/qecc/en/latest", None), + "syrec": ("https://mqt.readthedocs.io/projects/syrec/en/latest", None), +} + +nbsphinx_execute = "auto" +highlight_language = "python3" +nbsphinx_execute_arguments = [ + "--InlineBackend.figure_formats={'svg', 'pdf'}", + "--InlineBackend.rc=figure.dpi=200", +] +nbsphinx_kernel_name = "python3" + +autosectionlabel_prefix_document = True + +hoverxref_auto_ref = True +hoverxref_domains = ["cite", "py"] +hoverxref_roles = [] +hoverxref_mathjax = True +hoverxref_role_types = { + "ref": "tooltip", + "p": "tooltip", + "labelpar": "tooltip", + "class": "tooltip", + "meth": "tooltip", + "func": "tooltip", + "attr": "tooltip", + "property": "tooltip", +} +exclude_patterns = ["_build", "build", "**.ipynb_checkpoints", "Thumbs.db", ".DS_Store", ".env"] + + +class CDAStyle(UnsrtStyle): + """Custom style for including PDF links.""" + + def format_url(self, _e: Entry) -> HRef: # noqa: PLR6301 + """Format URL field as a link to the PDF.""" + url = field("url", raw=True) + return href()[url, "[PDF]"] + + +pybtex.plugin.register_plugin("pybtex.style.formatting", "cda_style", CDAStyle) + +bibtex_bibfiles = ["refs.bib"] +bibtex_default_style = "cda_style" + +copybutton_prompt_text = r"(?:\(venv\) )?(?:\[.*\] )?\$ " +copybutton_prompt_is_regexp = True +copybutton_line_continuation_character = "\\" + +autosummary_generate = True + +typehints_use_rtype = False +napoleon_use_rtype = False +napoleon_google_docstring = True +napoleon_numpy_docstring = False + +# -- Options for HTML output ------------------------------------------------- +html_theme = "furo" +html_static_path = ["_static"] +html_theme_options = { + "light_logo": "mqt_dark.png", + "dark_logo": "mqt_light.png", + "source_repository": "https://github.com/cda-tum/mqt-debugger/", + "source_branch": "main", + "source_directory": "docs/source", + "navigation_with_keys": True, +} diff --git a/docs/source/index.rst b/docs/source/index.rst new file mode 100644 index 0000000..c96b8a4 --- /dev/null +++ b/docs/source/index.rst @@ -0,0 +1,50 @@ +Welcome to MQT Debugger's documentation! +======================================== + +MQT Debugger is a comprehensive framework for debugging and analyzing the behaviour of quantum programs. It is developed by the `Chair for Design Automation `_ at the `Technical University of Munich `_ as part of the :doc:`Munich Quantum Toolkit ` (*MQT*). + +The framework provides tools for running quantum programs in a simulated environment, stepping through the code instruction-by-instruction, and displaying the quantum state at each step. +It also allows developers to include assertions in their code, testing the correctness of provided programs, and, when an assertion fails, suggesting possible causes of the failure. + +MQT Debugger is accessible as a stand-alone application, as a C++ and Python library to include in custom programs, and as a DAP server that can be accessed by popular IDEs such as Visual Studio Code and CLion. + +We recommend you to start with the :doc:`installation instructions `. +Then proceed to the :doc:`quickstart guide ` and read the :doc:`reference documentation `. +If you are interested in the theory behind MQT Debugger, have a look at the publications in the :doc:`publication list `. + +We appreciate any feedback and contributions to the project. If you want to contribute, you can find more information in the :doc:`Contribution ` guide. If you are having trouble with the installation or the usage of QCEC, please let us know at our :doc:`Support ` page. + +---- + + .. toctree:: + :hidden: + + self + + .. toctree:: + :maxdepth: 2 + :caption: User Guide + :glob: + + Installation + Quickstart + Debugging + Assertions + Diagnosis + Publications + + .. toctree:: + :maxdepth: 2 + :caption: Developers + :glob: + + Contributing + DevelopmentGuide + Support + + .. toctree:: + :maxdepth: 6 + :caption: API Reference + :glob: + + library/Library diff --git a/docs/source/library/Library.rst b/docs/source/library/Library.rst new file mode 100644 index 0000000..d7c1953 --- /dev/null +++ b/docs/source/library/Library.rst @@ -0,0 +1,10 @@ +Library +======= + + .. toctree:: + :maxdepth: 4 + + interface/Interface + dd/Dd + parsing/Parsing + python/Python diff --git a/docs/source/library/dd/DDSimDebug.rst b/docs/source/library/dd/DDSimDebug.rst new file mode 100644 index 0000000..264ed04 --- /dev/null +++ b/docs/source/library/dd/DDSimDebug.rst @@ -0,0 +1,5 @@ +DDSimDebug.hpp +============== + +.. doxygenfile:: DDSimDebug.hpp + :project: mqt-debugger diff --git a/docs/source/library/dd/DDSimDiagnostics.rst b/docs/source/library/dd/DDSimDiagnostics.rst new file mode 100644 index 0000000..f73241d --- /dev/null +++ b/docs/source/library/dd/DDSimDiagnostics.rst @@ -0,0 +1,5 @@ +DDSimDiagnostics.hpp +==================== + +.. doxygenfile:: DDSimDiagnostics.hpp + :project: mqt-debugger diff --git a/docs/source/library/dd/Dd.rst b/docs/source/library/dd/Dd.rst new file mode 100644 index 0000000..de2bf6c --- /dev/null +++ b/docs/source/library/dd/Dd.rst @@ -0,0 +1,11 @@ +DD Implementation +================= + +This section documents the implementation of the C interfaces specifically using the DD package from `MQT Core `_ +as a simulation backend and OpenQASM as the input language. + + .. toctree:: + :maxdepth: 4 + + DDSimDebug + DDSimDiagnostics diff --git a/docs/source/library/interface/Common.rst b/docs/source/library/interface/Common.rst new file mode 100644 index 0000000..f177ffe --- /dev/null +++ b/docs/source/library/interface/Common.rst @@ -0,0 +1,5 @@ +common.h +======== + +.. doxygenfile:: common.h + :project: mqt-debugger diff --git a/docs/source/library/interface/Debug.rst b/docs/source/library/interface/Debug.rst new file mode 100644 index 0000000..6261510 --- /dev/null +++ b/docs/source/library/interface/Debug.rst @@ -0,0 +1,10 @@ +debug.h +======= + +.. note:: + + As the `SimulationState` interface is defined in C, "member functions" are declared as function pointers. + When using the interface, these function pointers can be accessed like normal C++ methods. + +.. doxygenfile:: debug.h + :project: mqt-debugger diff --git a/docs/source/library/interface/Diagnostics.rst b/docs/source/library/interface/Diagnostics.rst new file mode 100644 index 0000000..2c421ac --- /dev/null +++ b/docs/source/library/interface/Diagnostics.rst @@ -0,0 +1,10 @@ +diagnostics.h +============= + +.. note:: + + As the `Diagnostics` interface is defined in C, "member functions" are declared as function pointers. + When using the interface, these function pointers can be accessed like normal C++ methods. + +.. doxygenfile:: diagnostics.h + :project: mqt-debugger diff --git a/docs/source/library/interface/Interface.rst b/docs/source/library/interface/Interface.rst new file mode 100644 index 0000000..1e7fe4e --- /dev/null +++ b/docs/source/library/interface/Interface.rst @@ -0,0 +1,16 @@ +C Interface +=========== + +This section documents the interface that must be implemented for a debugger to be compatible with this framework. + +The interface is written in C and can be implemented with any compatible language. We provide an :doc:`example implementation <../dd/Dd>` in C++ +based on the DD simulation backend from `MQT Core `_. + + .. toctree:: + :maxdepth: 4 + :caption: C-Interfaces + :glob: + + Common + Debug + Diagnostics diff --git a/docs/source/library/parsing/AssertionParsing.rst b/docs/source/library/parsing/AssertionParsing.rst new file mode 100644 index 0000000..f2ffd91 --- /dev/null +++ b/docs/source/library/parsing/AssertionParsing.rst @@ -0,0 +1,5 @@ +AssertionParsing.hpp +===================== + +.. doxygenfile:: AssertionParsing.hpp + :project: mqt-debugger diff --git a/docs/source/library/parsing/CodePreprocessing.rst b/docs/source/library/parsing/CodePreprocessing.rst new file mode 100644 index 0000000..fe29b5c --- /dev/null +++ b/docs/source/library/parsing/CodePreprocessing.rst @@ -0,0 +1,5 @@ +CodePreprocessing.hpp +===================== + +.. doxygenfile:: CodePreprocessing.hpp + :project: mqt-debugger diff --git a/docs/source/library/parsing/Parsing.rst b/docs/source/library/parsing/Parsing.rst new file mode 100644 index 0000000..de08cdb --- /dev/null +++ b/docs/source/library/parsing/Parsing.rst @@ -0,0 +1,13 @@ +Parsing +======= + +This section documents the parsing and preprocessing functionalities developed for this framework, partially for the :doc:`DD implementation <../dd/Dd>` of the debugger, +and partially for general-purpose use. + + .. toctree:: + :maxdepth: 4 + + AssertionParsing + CodePreprocessing + ParsingError + Utils diff --git a/docs/source/library/parsing/ParsingError.rst b/docs/source/library/parsing/ParsingError.rst new file mode 100644 index 0000000..2e3b673 --- /dev/null +++ b/docs/source/library/parsing/ParsingError.rst @@ -0,0 +1,5 @@ +ParsingError.hpp +===================== + +.. doxygenfile:: ParsingError.hpp + :project: mqt-debugger diff --git a/docs/source/library/parsing/Utils.rst b/docs/source/library/parsing/Utils.rst new file mode 100644 index 0000000..1deb33b --- /dev/null +++ b/docs/source/library/parsing/Utils.rst @@ -0,0 +1,5 @@ +Utils.hpp +===================== + +.. doxygenfile:: Utils.hpp + :project: mqt-debugger diff --git a/docs/source/library/python/Debug.rst b/docs/source/library/python/Debug.rst new file mode 100644 index 0000000..1cb594a --- /dev/null +++ b/docs/source/library/python/Debug.rst @@ -0,0 +1,5 @@ +mqt.debugger +============ + +.. automodule:: mqt.debugger + :members: diff --git a/docs/source/library/python/Python.rst b/docs/source/library/python/Python.rst new file mode 100644 index 0000000..e8bfcee --- /dev/null +++ b/docs/source/library/python/Python.rst @@ -0,0 +1,10 @@ +Python +====== + +This section documents the Python modules that are provided by this framework. + + .. toctree:: + :maxdepth: 4 + + Debug + dap/DAP diff --git a/docs/source/library/python/dap/Adapter.rst b/docs/source/library/python/dap/Adapter.rst new file mode 100644 index 0000000..444e3f0 --- /dev/null +++ b/docs/source/library/python/dap/Adapter.rst @@ -0,0 +1,16 @@ +mqt.debugger.dap.adapter +============================ + +.. automodule:: mqt.debugger.dap.adapter + :members: + :undoc-members: + :show-inheritance: + + +Starting this module directly using + +.. code-block:: console + + python -m mqt.debugger.dap.adapter + +will start the Adapter server on the default port 4711. diff --git a/docs/source/library/python/dap/DAP.rst b/docs/source/library/python/dap/DAP.rst new file mode 100644 index 0000000..ef4e2c4 --- /dev/null +++ b/docs/source/library/python/dap/DAP.rst @@ -0,0 +1,13 @@ +mqt.debugger.dap +================= + +.. automodule:: mqt.debugger.dap + :members: + :undoc-members: + :show-inheritance: + +.. toctree:: + :maxdepth: 4 + + Adapter + Messages diff --git a/docs/source/library/python/dap/Messages.rst b/docs/source/library/python/dap/Messages.rst new file mode 100644 index 0000000..5f45cd1 --- /dev/null +++ b/docs/source/library/python/dap/Messages.rst @@ -0,0 +1,7 @@ +mqt.debugger.dap.messages +========================= + +.. automodule:: mqt.debugger.dap.messages + :members: + :undoc-members: + :show-inheritance: diff --git a/docs/source/refs.bib b/docs/source/refs.bib new file mode 100644 index 0000000..e69de29 diff --git a/include/backend/dd/DDSimDebug.hpp b/include/backend/dd/DDSimDebug.hpp index 7a06edf..4a51ae4 100644 --- a/include/backend/dd/DDSimDebug.hpp +++ b/include/backend/dd/DDSimDebug.hpp @@ -434,11 +434,11 @@ size_t ddsimGetNumQubits(SimulationState* self); * representation of the state. * * @param self The instance to query. - * @param qubit The index of the qubit. + * @param index The index of the state. * @param output A reference to a `Complex` instance to store the amplitude. * @return The result of the operation. */ -Result ddsimGetAmplitudeIndex(SimulationState* self, size_t qubit, +Result ddsimGetAmplitudeIndex(SimulationState* self, size_t index, Complex* output); /** * @brief Gets the complex amplitude of a state in the full state vector. @@ -446,7 +446,7 @@ Result ddsimGetAmplitudeIndex(SimulationState* self, size_t qubit, * The amplitude is selected by a bitstring representing the state. * * @param self The instance to query. - * @param qubit The index of the qubit as a bitstring. + * @param bitstring The index of the qubit as a bitstring. * @param output A reference to a `Complex` instance to store the amplitude. * @return The result of the operation. */ diff --git a/include/backend/debug.h b/include/backend/debug.h index ded0a4c..cb12ce5 100644 --- a/include/backend/debug.h +++ b/include/backend/debug.h @@ -15,11 +15,6 @@ extern "C" { #endif -/** - * @brief A C-style interface for the debugging and simulation interface. - */ -typedef struct SimulationState SimulationState; - /** * @brief A C-style interface for the debugging and simulation interface. * @@ -31,7 +26,9 @@ typedef struct SimulationState SimulationState; * Continuing the simulation from there will skip the failing assertion and * `didAssertionFail` will return `false` until another assertion fails. */ -struct SimulationState { +typedef struct SimulationStateStruct SimulationState; + +struct SimulationStateStruct { /** * @brief Initializes the simulation state. * @param self The instance to initialize. @@ -168,11 +165,11 @@ struct SimulationState { bool (*canStepBackward)(SimulationState* self); /** - * @brief Indicates whether the simulation has finished. + * @brief Indicates whether the execution has finished. * - * The simulation is considered finished if it has reached the end of the code + * The execution is considered finished if it has reached the end of the code. * @param self The instance to query. - * @return True if the simulation has finished, false otherwise. + * @return True if the execution has finished, false otherwise. */ bool (*isFinished)(SimulationState* self); @@ -237,11 +234,11 @@ struct SimulationState { * binary representation of the state. * * @param self The instance to query. - * @param qubit The index of the qubit. + * @param index The index of the state. * @param output A reference to a `Complex` instance to store the amplitude. * @return The result of the operation. */ - Result (*getAmplitudeIndex)(SimulationState* self, size_t qubit, + Result (*getAmplitudeIndex)(SimulationState* self, size_t index, Complex* output); /** @@ -250,7 +247,7 @@ struct SimulationState { * The amplitude is selected by a bitstring representing the state. * * @param self The instance to query. - * @param qubit The index of the qubit as a bitstring. + * @param bitstring The index of the state as a bitstring. * @param output A reference to a `Complex` instance to store the amplitude. * @return The result of the operation. */ diff --git a/include/backend/diagnostics.h b/include/backend/diagnostics.h index de39d92..36400d5 100644 --- a/include/backend/diagnostics.h +++ b/include/backend/diagnostics.h @@ -40,8 +40,9 @@ typedef enum { /** * @brief Represents an error cause. */ -typedef struct ErrorCause ErrorCause; -struct ErrorCause { +typedef struct ErrorCauseStruct ErrorCause; + +struct ErrorCauseStruct { /** * @brief The type of the error cause. */ @@ -56,11 +57,9 @@ struct ErrorCause { /** * @brief An interface representing the diagnostic capabilities of a debugger. */ -typedef struct Diagnostics Diagnostics; -/** - * @brief An interface representing the diagnostic capabilities of a debugger. - */ -struct Diagnostics { +typedef struct DiagnosticsStruct Diagnostics; + +struct DiagnosticsStruct { /** * @brief Initializes the diagnostics interface. * @param self The instance to initialize. @@ -103,7 +102,7 @@ struct Diagnostics { * * @param self The diagnostics instance to query. * @param instruction The instruction to extract the data dependencies for. - * @param includeCallers True if the data dependencies should include all + * @param includeCallers True, if the data dependencies should include all * possible callers of the containing custom gate. * @param instructions An array of booleans that will be set to true for each * instruction that is a data dependency. diff --git a/include/common.h b/include/common.h index 7e21223..0954132 100644 --- a/include/common.h +++ b/include/common.h @@ -10,7 +10,6 @@ // NOLINTBEGIN(modernize-use-using, performance-enum-size) #ifndef __cplusplus -typedef char bool; #else #include extern "C" { diff --git a/pyproject.toml b/pyproject.toml index af60f1e..4f79387 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -3,7 +3,7 @@ requires = ["scikit-build-core>=0.8.1", "setuptools-scm>=7", "pybind11>=2.13"] build-backend = "scikit_build_core.build" [project] -name = "mqt-debug" +name = "mqt-debugger" description = "A quantum circuit debugging tool" readme = "README.md" authors = [ @@ -36,8 +36,8 @@ requires-python = ">=3.8" dynamic = ["version"] [project.optional-dependencies] -test = ["pytest>=7.0", "pytest-console-scripts>=1.4", "mqt.debug[qiskit, evaluation]"] -coverage = ["mqt.debug[test]", "pytest-cov>=4"] +test = ["pytest>=7.0", "pytest-console-scripts>=1.4", "mqt.debugger[qiskit, evaluation]"] +coverage = ["mqt.debugger[test]", "pytest-cov>=4"] evaluation = [ "pandas[output_formatting]>=2.0; python_version < '3.9'", "pandas[output-formatting]>=2.1.2; python_version >= '3.9'", @@ -57,21 +57,21 @@ docs = [ "ipykernel", "sphinx-autoapi", "qiskit[visualization]", - "mqt.debug[evaluation]" + "mqt.debugger[evaluation]" ] qiskit = [ "qiskit[qasm3-import]>=1.0.0", ] -dev = ["mqt.debug[coverage, docs]",] +dev = ["mqt.debugger[coverage, docs]",] [project.scripts] -mqt-debug-compare = "mqt.debug.evaluation:main" +mqt-debugger-compare = "mqt.debugger.evaluation:main" [project.urls] -Homepage = "https://github.com/cda-tum/mqt-debug" -Documentation = "https://mqt.readthedocs.io/projects/debug" -Issues = "https://github.com/cda-tum/mqt-debug/issues" -Discussions = "https://github.com/cda-tum/mqt-debug/discussions" +Homepage = "https://github.com/cda-tum/mqt-debugger" +Documentation = "https://mqt.readthedocs.io/projects/debugger" +Issues = "https://github.com/cda-tum/mqt-debugger/issues" +Discussions = "https://github.com/cda-tum/mqt-debugger/discussions" [tool.scikit-build] @@ -79,7 +79,7 @@ Discussions = "https://github.com/cda-tum/mqt-debug/discussions" minimum-version = "0.8.1" # Set the wheel install directory -wheel.install-dir = "mqt/debug" +wheel.install-dir = "mqt/debugger" # Set required CMake and Ninja versions cmake.version = ">=3.19" @@ -92,13 +92,13 @@ build-dir = "build/{wheel_tag}" wheel.packages = ["src/mqt"] # Only build the Python bindings target -cmake.targets = ["pydebug"] +cmake.targets = ["pydebugger"] # Only install the Python package component -install.components = ["mqt-debug_Python"] +install.components = ["mqt-debugger_Python"] metadata.version.provider = "scikit_build_core.metadata.setuptools_scm" -sdist.include = ["src/mqt/debug/_version.py"] +sdist.include = ["src/mqt/debugger/_version.py"] sdist.exclude = [ "**/.github", "**/doc", @@ -110,19 +110,20 @@ sdist.exclude = [ ] [tool.check-sdist] -sdist-only = ["src/mqt/debug/_version.py"] +sdist-only = ["src/mqt/debugger/_version.py"] git-only = [ "docs/*", "test/*", ] [tool.scikit-build.cmake.define] -BUILD_MQT_DEBUG_BINDINGS = "ON" -BUILD_MQT_DEBUG_TESTS = "OFF" +BUILD_MQT_DEBUGGER_BINDINGS = "ON" +BUILD_MQT_DEBUGGER_TESTS = "OFF" +BUILD_MQT_DEBUGGER_APP = "OFF" [tool.setuptools_scm] -write_to = "src/mqt/debug/_version.py" +write_to = "src/mqt/debugger/_version.py" [tool.pytest.ini_options] @@ -140,7 +141,7 @@ testpaths = ["test/python"] [tool.coverage] -run.source = ["mqt.debug"] +run.source = ["mqt.debugger"] run.omit = [ '*/_compat/*', ] @@ -228,7 +229,7 @@ ignore = [ "S101", # Use of assert detected "S404", # `subprocess` module is possibly insecure ] -typing-modules = ["mqt.debug._compat.typing"] +typing-modules = ["mqt.debugger._compat.typing"] isort.required-imports = ["from __future__ import annotations"] [tool.ruff.lint.flake8-tidy-imports.banned-api] @@ -237,9 +238,9 @@ isort.required-imports = ["from __future__ import annotations"] "typing.Mapping".msg = "Use collections.abc.Mapping instead." "typing.Sequence".msg = "Use collections.abc.Sequence instead." "typing.Set".msg = "Use collections.abc.Set instead." -"typing.Self".msg = "Use mqt.debug._compat.typing.Self instead." -"typing_extensions.Self".msg = "Use mqt.debug._compat.typing.Self instead." -"typing.assert_never".msg = "Use mqt.debug._compat.typing.assert_never instead." +"typing.Self".msg = "Use mqt.debugger._compat.typing.Self instead." +"typing_extensions.Self".msg = "Use mqt.debugger._compat.typing.Self instead." +"typing.assert_never".msg = "Use mqt.debugger._compat.typing.assert_never instead." [tool.ruff.lint.per-file-ignores] "test/python/**" = ["T20", "ANN"] @@ -251,9 +252,9 @@ isort.required-imports = ["from __future__ import annotations"] "E402", # Allow imports to appear anywhere in Jupyter notebooks "I002", # Allow missing `from __future__ import annotations` import ] -"src/mqt/debug/_compat/**.py" = ["TID251"] -"src/mqt/debug/evaluation.py" = ["T201"] -"src/mqt/debug/__main__.py" = ["T201"] +"src/mqt/debugger/_compat/**.py" = ["TID251"] +"src/mqt/debugger/evaluation.py" = ["T201"] +"src/mqt/debugger/__main__.py" = ["T201"] [tool.ruff.lint.pydocstyle] convention = "google" @@ -276,7 +277,7 @@ ignore = [ build = "cp3*" skip = "*-musllinux_*" archs = "auto64" -test-command = "python -c \"from mqt import debug\"" +test-command = "python -c \"from mqt import debugger\"" test-skip = "cp38-macosx_arm64" build-frontend = "build[uv]" @@ -289,4 +290,3 @@ environment = { MACOSX_DEPLOYMENT_TARGET = "10.15" } [tool.cibuildwheel.windows] before-build = "uv pip install delvewheel>=1.4.0" repair-wheel-command = "delvewheel repair -w {dest_dir} {wheel} --namespace-pkg mqt" -environment = { CMAKE_GENERATOR = "Ninja" } diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt index abb9c7b..8d56e88 100644 --- a/src/CMakeLists.txt +++ b/src/CMakeLists.txt @@ -18,8 +18,8 @@ target_link_libraries(${PROJECT_NAME} PRIVATE MQT::ProjectWarnings MQT::ProjectO target_link_libraries(${PROJECT_NAME} PRIVATE Eigen3::Eigen) # add MQT alias -add_library(MQT::Debug ALIAS ${PROJECT_NAME}) +add_library(MQT::Debugger ALIAS ${PROJECT_NAME}) -if(BUILD_MQT_DEBUG_BINDINGS) +if(BUILD_MQT_DEBUGGER_BINDINGS) add_subdirectory(python) endif() diff --git a/src/backend/dd/DDSimDebug.cpp b/src/backend/dd/DDSimDebug.cpp index 7de4f09..a08b174 100644 --- a/src/backend/dd/DDSimDebug.cpp +++ b/src/backend/dd/DDSimDebug.cpp @@ -665,10 +665,10 @@ size_t ddsimGetNumQubits(SimulationState* self) { return ddsim->qc->getNqubits(); } -Result ddsimGetAmplitudeIndex(SimulationState* self, size_t qubit, +Result ddsimGetAmplitudeIndex(SimulationState* self, size_t index, Complex* output) { auto* ddsim = toDDSimulationState(self); - auto result = ddsim->simulationState.getValueByIndex(qubit); + auto result = ddsim->simulationState.getValueByIndex(index); output->real = result.real(); output->imaginary = result.imag(); return OK; diff --git a/src/mqt/debug/pydebug.pyi b/src/mqt/debug/pydebug.pyi deleted file mode 100644 index f6235ea..0000000 --- a/src/mqt/debug/pydebug.pyi +++ /dev/null @@ -1,98 +0,0 @@ -"""Type stubs for python bindings of the debug module.""" - -# Enums - -class VariableType: - VarBool: VariableType - VarInt: VariableType - VarFloat: VariableType - -# Classes -class VariableValue: - bool_value: bool - int_value: int - float_value: float - - def __init__(self) -> None: ... - -class Variable: - name: str - type: VariableType - value: VariableValue - - def __init__(self) -> None: ... - -class Complex: - real: float - imaginary: float - - def __init__(self, real: float = 0.0, imaginary: float = 0.0) -> None: ... - -class Statevector: - num_qubits: int - num_states: int - amplitudes: list[Complex] - - def __init__(self) -> None: ... - -class SimulationState: - def __init__(self) -> None: ... - def init(self) -> None: ... - def load_code(self, code: str) -> None: ... - def step_forward(self) -> None: ... - def step_over_forward(self) -> None: ... - def step_out_forward(self) -> None: ... - def step_backward(self) -> None: ... - def step_over_backward(self) -> None: ... - def step_out_backward(self) -> None: ... - def run_all(self) -> int: ... - def run_simulation(self) -> None: ... - def run_simulation_backward(self) -> None: ... - def reset_simulation(self) -> None: ... - def pause_simulation(self) -> None: ... - def can_step_forward(self) -> bool: ... - def can_step_backward(self) -> bool: ... - def is_finished(self) -> bool: ... - def did_assertion_fail(self) -> bool: ... - def was_breakpoint_hit(self) -> bool: ... - def get_current_instruction(self) -> int: ... - def get_previous_instruction(self) -> int: ... - def get_instruction_count(self) -> int: ... - def get_instruction_position(self, instruction: int) -> tuple[int, int]: ... - def get_num_qubits(self) -> int: ... - def get_amplitude_index(self, qubit: int) -> Complex: ... - def get_amplitude_bitstring(self, bitstring: str) -> Complex: ... - def get_classical_variable(self, name: str) -> Variable: ... - def get_num_classical_variables(self) -> int: ... - def get_classical_variable_name(self, index: int) -> str: ... - def get_state_vector_full(self) -> Statevector: ... - def get_state_vector_sub(self, qubits: list[int]) -> Statevector: ... - def set_breakpoint(self, desired_position: int) -> int: ... - def clear_breakpoints(self) -> None: ... - def get_stack_depth(self) -> int: ... - def get_stack_trace(self, max_depth: int) -> list[int]: ... - def get_diagnostics(self) -> Diagnostics: ... - -class ErrorCauseType: - Unknown: ErrorCauseType - MissingInteraction: ErrorCauseType - ControlAlwaysZero: ErrorCauseType - -class ErrorCause: - instruction: int - type: ErrorCauseType - - def __init__(self) -> None: ... - -class Diagnostics: - def __init__(self) -> None: ... - def init(self) -> None: ... - def get_num_qubits(self) -> int: ... - def get_instruction_count(self) -> int: ... - def get_data_dependencies(self, instruction: int, include_callers: bool = False) -> list[int]: ... - def get_interactions(self, before_instruction: int, qubit: int) -> list[int]: ... - def get_zero_control_instructions(self) -> list[int]: ... - def potential_error_causes(self) -> list[ErrorCause]: ... - -def create_ddsim_simulation_state() -> SimulationState: ... -def destroy_ddsim_simulation_state(state: SimulationState) -> None: ... diff --git a/src/mqt/debug/__init__.py b/src/mqt/debugger/__init__.py similarity index 73% rename from src/mqt/debug/__init__.py rename to src/mqt/debugger/__init__.py index db99109..7f2c61d 100644 --- a/src/mqt/debug/__init__.py +++ b/src/mqt/debugger/__init__.py @@ -1,10 +1,12 @@ -"""A module for interfacing with the mqt-debug library.""" +"""A module for interfacing with the mqt-debugger library.""" from __future__ import annotations from . import dap -from .pydebug import ( +from ._version import version as __version__ +from .pydebugger import ( Complex, + Diagnostics, ErrorCause, ErrorCauseType, SimulationState, @@ -18,6 +20,7 @@ __all__ = [ "Complex", + "Diagnostics", "ErrorCause", "ErrorCauseType", "SimulationState", @@ -25,6 +28,7 @@ "Variable", "VariableType", "VariableValue", + "__version__", "create_ddsim_simulation_state", "dap", "destroy_ddsim_simulation_state", diff --git a/src/mqt/debugger/_version.pyi b/src/mqt/debugger/_version.pyi new file mode 100644 index 0000000..9312dff --- /dev/null +++ b/src/mqt/debugger/_version.pyi @@ -0,0 +1,4 @@ +__version__: str +version: str +__version_tuple__: tuple[int, int, int, str, str] | tuple[int, int, int] +version_tuple: tuple[int, int, int, str, str] | tuple[int, int, int] diff --git a/src/mqt/debug/dap/__init__.py b/src/mqt/debugger/dap/__init__.py similarity index 100% rename from src/mqt/debug/dap/__init__.py rename to src/mqt/debugger/dap/__init__.py diff --git a/src/mqt/debug/dap/adapter.py b/src/mqt/debugger/dap/adapter.py similarity index 100% rename from src/mqt/debug/dap/adapter.py rename to src/mqt/debugger/dap/adapter.py diff --git a/src/mqt/debug/dap/dap_server.py b/src/mqt/debugger/dap/dap_server.py similarity index 81% rename from src/mqt/debug/dap/dap_server.py rename to src/mqt/debugger/dap/dap_server.py index eb6990c..2646b12 100644 --- a/src/mqt/debug/dap/dap_server.py +++ b/src/mqt/debugger/dap/dap_server.py @@ -7,9 +7,8 @@ import sys from typing import Any, cast -import mqt.debug +import mqt.debugger -from ..pydebug import ErrorCauseType, SimulationState from .messages import ( ConfigurationDoneDAPMessage, ContinueDAPMessage, @@ -79,7 +78,7 @@ class DAPServer: host: str port: int - simulation_state: SimulationState + simulation_state: mqt.debugger.SimulationState source_file: dict[str, Any] source_code: str can_step_back: bool @@ -97,7 +96,7 @@ def __init__(self, host: str = "127.0.0.1", port: int = 4711) -> None: self.host = host self.port = port self.can_step_back = False - self.simulation_state = SimulationState() + self.simulation_state = mqt.debugger.SimulationState() self.lines_start_at_one = True self.columns_start_at_one = True @@ -153,45 +152,49 @@ def handle_client(self, connection: socket.socket) -> None: result_payload = json.dumps(result) send_message(result_payload, connection) - e: mqt.debug.dap.messages.DAPEvent | None = None - if isinstance(cmd, mqt.debug.dap.messages.LaunchDAPMessage): - e = mqt.debug.dap.messages.InitializedDAPEvent() + e: mqt.debugger.dap.messages.DAPEvent | None = None + if isinstance(cmd, mqt.debugger.dap.messages.LaunchDAPMessage): + e = mqt.debugger.dap.messages.InitializedDAPEvent() event_payload = json.dumps(e.encode()) send_message(event_payload, connection) if ( - isinstance(cmd, (mqt.debug.dap.messages.LaunchDAPMessage, mqt.debug.dap.messages.RestartDAPMessage)) + isinstance( + cmd, (mqt.debugger.dap.messages.LaunchDAPMessage, mqt.debugger.dap.messages.RestartDAPMessage) + ) and cmd.stop_on_entry ): - e = mqt.debug.dap.messages.StoppedDAPEvent(mqt.debug.dap.messages.StopReason.ENTRY, "Stopped on entry") + e = mqt.debugger.dap.messages.StoppedDAPEvent( + mqt.debugger.dap.messages.StopReason.ENTRY, "Stopped on entry" + ) event_payload = json.dumps(e.encode()) send_message(event_payload, connection) if isinstance( cmd, ( - mqt.debug.dap.messages.NextDAPMessage, - mqt.debug.dap.messages.StepBackDAPMessage, - mqt.debug.dap.messages.StepInDAPMessage, - mqt.debug.dap.messages.StepOutDAPMessage, - mqt.debug.dap.messages.ContinueDAPMessage, - mqt.debug.dap.messages.ReverseContinueDAPMessage, - mqt.debug.dap.messages.RestartFrameDAPMessage, + mqt.debugger.dap.messages.NextDAPMessage, + mqt.debugger.dap.messages.StepBackDAPMessage, + mqt.debugger.dap.messages.StepInDAPMessage, + mqt.debugger.dap.messages.StepOutDAPMessage, + mqt.debugger.dap.messages.ContinueDAPMessage, + mqt.debugger.dap.messages.ReverseContinueDAPMessage, + mqt.debugger.dap.messages.RestartFrameDAPMessage, ), ) or ( isinstance( cmd, ( - mqt.debug.dap.messages.LaunchDAPMessage, - mqt.debug.dap.messages.RestartDAPMessage, + mqt.debugger.dap.messages.LaunchDAPMessage, + mqt.debugger.dap.messages.RestartDAPMessage, ), ) and not cmd.stop_on_entry ): event = ( - mqt.debug.dap.messages.StopReason.EXCEPTION + mqt.debugger.dap.messages.StopReason.EXCEPTION if self.simulation_state.did_assertion_fail() - else mqt.debug.dap.messages.StopReason.BREAKPOINT_INSTRUCTION + else mqt.debugger.dap.messages.StopReason.BREAKPOINT_INSTRUCTION if self.simulation_state.was_breakpoint_hit() - else mqt.debug.dap.messages.StopReason.STEP + else mqt.debugger.dap.messages.StopReason.STEP ) message = ( "An assertion failed" @@ -200,21 +203,21 @@ def handle_client(self, connection: socket.socket) -> None: if self.simulation_state.was_breakpoint_hit() else "Stopped after step" ) - e = mqt.debug.dap.messages.StoppedDAPEvent(event, message) + e = mqt.debugger.dap.messages.StoppedDAPEvent(event, message) event_payload = json.dumps(e.encode()) send_message(event_payload, connection) if self.simulation_state.did_assertion_fail(): self.handle_assertion_fail(connection) - if isinstance(cmd, mqt.debug.dap.messages.TerminateDAPMessage): - e = mqt.debug.dap.messages.TerminatedDAPEvent() + if isinstance(cmd, mqt.debugger.dap.messages.TerminateDAPMessage): + e = mqt.debugger.dap.messages.TerminatedDAPEvent() event_payload = json.dumps(e.encode()) send_message(event_payload, connection) - e = mqt.debug.dap.messages.ExitedDAPEvent(143) + e = mqt.debugger.dap.messages.ExitedDAPEvent(143) event_payload = json.dumps(e.encode()) send_message(event_payload, connection) - if isinstance(cmd, mqt.debug.dap.messages.PauseDAPMessage): - e = mqt.debug.dap.messages.StoppedDAPEvent( - mqt.debug.dap.messages.StopReason.PAUSE, "Stopped after pause" + if isinstance(cmd, mqt.debugger.dap.messages.PauseDAPMessage): + e = mqt.debugger.dap.messages.StoppedDAPEvent( + mqt.debugger.dap.messages.StopReason.PAUSE, "Stopped after pause" ) event_payload = json.dumps(e.encode()) send_message(event_payload, connection) @@ -226,17 +229,17 @@ def regular_checks(self, connection: socket.socket) -> None: Args: connection (socket.socket): The client socket. """ - e: mqt.debug.dap.messages.DAPEvent | None = None + e: mqt.debugger.dap.messages.DAPEvent | None = None if self.simulation_state.is_finished() and self.simulation_state.get_instruction_count() != 0: - e = mqt.debug.dap.messages.ExitedDAPEvent(0) + e = mqt.debugger.dap.messages.ExitedDAPEvent(0) event_payload = json.dumps(e.encode()) send_message(event_payload, connection) if self.can_step_back != self.simulation_state.can_step_backward(): self.can_step_back = self.simulation_state.can_step_backward() - e = mqt.debug.dap.messages.CapabilitiesDAPEvent({"supportsStepBack": self.can_step_back}) + e = mqt.debugger.dap.messages.CapabilitiesDAPEvent({"supportsStepBack": self.can_step_back}) event_payload = json.dumps(e.encode()) - def handle_command(self, command: dict[str, Any]) -> tuple[dict[str, Any], mqt.debug.dap.messages.DAPMessage]: + def handle_command(self, command: dict[str, Any]) -> tuple[dict[str, Any], mqt.debugger.dap.messages.DAPMessage]: """Handle an incoming command from the client and return the corresponding response. Args: @@ -246,7 +249,7 @@ def handle_command(self, command: dict[str, Any]) -> tuple[dict[str, Any], mqt.d RuntimeError: If the command is not supported. Returns: - tuple[dict[str, Any], mqt.debug.dap.messages.DAPMessage]: The response to the message as a dictionary and the message object. + tuple[dict[str, Any], mqt.debugger.dap.messages.DAPMessage]: The response to the message as a dictionary and the message object. """ for message_type in supported_messages: if message_type.message_type_name == command["command"]: @@ -270,7 +273,7 @@ def handle_assertion_fail(self, connection: socket.socket) -> None: start, end = self.simulation_state.get_instruction_position(i) gray_out_areas.append((start, end)) - e = mqt.debug.dap.messages.GrayOutDAPEvent(gray_out_areas, self.source_file) + e = mqt.debugger.dap.messages.GrayOutDAPEvent(gray_out_areas, self.source_file) event_payload = json.dumps(e.encode()) send_message(event_payload, connection) @@ -345,11 +348,11 @@ def code_coordinates_to_pos(self, line: int, col: int) -> int: pos -= 1 return pos - def format_error_cause(self, cause: mqt.debug.ErrorCause) -> str: + def format_error_cause(self, cause: mqt.debugger.ErrorCause) -> str: """Format an error cause for output. Args: - cause (mqt.debug.ErrorCause): The error cause. + cause (mqt.debugger.ErrorCause): The error cause. Returns: str: The formatted error cause. @@ -358,9 +361,9 @@ def format_error_cause(self, cause: mqt.debug.ErrorCause) -> str: start_line, _ = self.code_pos_to_coordinates(start_pos) return ( "The qubits never interact with each other. Are you missing a CX gate?" - if cause.type == ErrorCauseType.MissingInteraction + if cause.type == mqt.debugger.ErrorCauseType.MissingInteraction else f"Control qubit is always zero in line {start_line}." - if cause.type == ErrorCauseType.ControlAlwaysZero + if cause.type == mqt.debugger.ErrorCauseType.ControlAlwaysZero else "" ) @@ -370,13 +373,13 @@ def send_message_hierarchy( """Send a hierarchy of messages to the client. Args: - message (dict[str, str | list[str], dict[str, Any]]): An object representing the message to send. Supported keys are `title`, `body`, `end`. + message (dict[str, str | list[str], dict[str, Any]]): An object representing the message to send. Supported keys are "title", "body", "end". line (int): The line number. column (int): The column number. connection (socket.socket): The client socket. """ if "title" in message: - title_event = mqt.debug.dap.messages.OutputDAPEvent( + title_event = mqt.debugger.dap.messages.OutputDAPEvent( "console", cast(str, message["title"]), "start", line, column, self.source_file ) send_message(json.dumps(title_event.encode()), connection) @@ -388,20 +391,20 @@ def send_message_hierarchy( if isinstance(msg, dict): self.send_message_hierarchy(msg, line, column, connection) else: - output_event = mqt.debug.dap.messages.OutputDAPEvent( + output_event = mqt.debugger.dap.messages.OutputDAPEvent( "console", msg, None, line, column, self.source_file ) send_message(json.dumps(output_event.encode()), connection) elif isinstance(body, dict): self.send_message_hierarchy(body, line, column, connection) elif isinstance(body, str): - output_event = mqt.debug.dap.messages.OutputDAPEvent( + output_event = mqt.debugger.dap.messages.OutputDAPEvent( "console", body, None, line, column, self.source_file ) send_message(json.dumps(output_event.encode()), connection) if "end" in message or "title" in message: - end_event = mqt.debug.dap.messages.OutputDAPEvent( + end_event = mqt.debugger.dap.messages.OutputDAPEvent( "console", cast(str, message.get("end")), "end", line, column, self.source_file ) send_message(json.dumps(end_event.encode()), connection) diff --git a/src/mqt/debug/dap/messages/__init__.py b/src/mqt/debugger/dap/messages/__init__.py similarity index 100% rename from src/mqt/debug/dap/messages/__init__.py rename to src/mqt/debugger/dap/messages/__init__.py diff --git a/src/mqt/debug/dap/messages/capabilities_dap_event.py b/src/mqt/debugger/dap/messages/capabilities_dap_event.py similarity index 100% rename from src/mqt/debug/dap/messages/capabilities_dap_event.py rename to src/mqt/debugger/dap/messages/capabilities_dap_event.py diff --git a/src/mqt/debug/dap/messages/configuration_done_dap_message.py b/src/mqt/debugger/dap/messages/configuration_done_dap_message.py similarity index 100% rename from src/mqt/debug/dap/messages/configuration_done_dap_message.py rename to src/mqt/debugger/dap/messages/configuration_done_dap_message.py diff --git a/src/mqt/debug/dap/messages/continue_dap_message.py b/src/mqt/debugger/dap/messages/continue_dap_message.py similarity index 100% rename from src/mqt/debug/dap/messages/continue_dap_message.py rename to src/mqt/debugger/dap/messages/continue_dap_message.py diff --git a/src/mqt/debug/dap/messages/dap_event.py b/src/mqt/debugger/dap/messages/dap_event.py similarity index 100% rename from src/mqt/debug/dap/messages/dap_event.py rename to src/mqt/debugger/dap/messages/dap_event.py diff --git a/src/mqt/debug/dap/messages/dap_message.py b/src/mqt/debugger/dap/messages/dap_message.py similarity index 100% rename from src/mqt/debug/dap/messages/dap_message.py rename to src/mqt/debugger/dap/messages/dap_message.py diff --git a/src/mqt/debug/dap/messages/disconnect_dap_message.py b/src/mqt/debugger/dap/messages/disconnect_dap_message.py similarity index 91% rename from src/mqt/debug/dap/messages/disconnect_dap_message.py rename to src/mqt/debugger/dap/messages/disconnect_dap_message.py index f8e89fb..b73558d 100644 --- a/src/mqt/debug/dap/messages/disconnect_dap_message.py +++ b/src/mqt/debugger/dap/messages/disconnect_dap_message.py @@ -4,7 +4,7 @@ from typing import TYPE_CHECKING, Any -import mqt.debug +import mqt.debugger from .dap_message import DAPMessage @@ -37,5 +37,5 @@ def handle(self, server: DAPServer) -> dict[str, Any]: Returns: dict[str, Any]: The response to the request. """ - mqt.debug.destroy_ddsim_simulation_state(server.simulation_state) + mqt.debugger.destroy_ddsim_simulation_state(server.simulation_state) return super().handle(server) diff --git a/src/mqt/debug/dap/messages/exception_info_message.py b/src/mqt/debugger/dap/messages/exception_info_message.py similarity index 100% rename from src/mqt/debug/dap/messages/exception_info_message.py rename to src/mqt/debugger/dap/messages/exception_info_message.py diff --git a/src/mqt/debug/dap/messages/exited_dap_event.py b/src/mqt/debugger/dap/messages/exited_dap_event.py similarity index 100% rename from src/mqt/debug/dap/messages/exited_dap_event.py rename to src/mqt/debugger/dap/messages/exited_dap_event.py diff --git a/src/mqt/debug/dap/messages/gray_out_event.py b/src/mqt/debugger/dap/messages/gray_out_event.py similarity index 100% rename from src/mqt/debug/dap/messages/gray_out_event.py rename to src/mqt/debugger/dap/messages/gray_out_event.py diff --git a/src/mqt/debug/dap/messages/initialize_dap_message.py b/src/mqt/debugger/dap/messages/initialize_dap_message.py similarity index 95% rename from src/mqt/debug/dap/messages/initialize_dap_message.py rename to src/mqt/debugger/dap/messages/initialize_dap_message.py index ea050e4..0dd1962 100644 --- a/src/mqt/debug/dap/messages/initialize_dap_message.py +++ b/src/mqt/debugger/dap/messages/initialize_dap_message.py @@ -4,7 +4,7 @@ from typing import TYPE_CHECKING, Any -import mqt.debug +import mqt.debugger from .dap_message import DAPMessage from .utils import get_default_capabilities @@ -60,7 +60,7 @@ def handle(self, server: DAPServer) -> dict[str, Any]: """ server.columns_start_at_one = self.columns_start_at1 server.lines_start_at_one = self.lines_start_at1 - server.simulation_state = mqt.debug.create_ddsim_simulation_state() + server.simulation_state = mqt.debugger.create_ddsim_simulation_state() return { "type": "response", "request_seq": self.sequence_number, diff --git a/src/mqt/debug/dap/messages/initialized_dap_event.py b/src/mqt/debugger/dap/messages/initialized_dap_event.py similarity index 100% rename from src/mqt/debug/dap/messages/initialized_dap_event.py rename to src/mqt/debugger/dap/messages/initialized_dap_event.py diff --git a/src/mqt/debug/dap/messages/launch_dap_message.py b/src/mqt/debugger/dap/messages/launch_dap_message.py similarity index 100% rename from src/mqt/debug/dap/messages/launch_dap_message.py rename to src/mqt/debugger/dap/messages/launch_dap_message.py diff --git a/src/mqt/debug/dap/messages/next_dap_message.py b/src/mqt/debugger/dap/messages/next_dap_message.py similarity index 100% rename from src/mqt/debug/dap/messages/next_dap_message.py rename to src/mqt/debugger/dap/messages/next_dap_message.py diff --git a/src/mqt/debug/dap/messages/output_dap_event.py b/src/mqt/debugger/dap/messages/output_dap_event.py similarity index 100% rename from src/mqt/debug/dap/messages/output_dap_event.py rename to src/mqt/debugger/dap/messages/output_dap_event.py diff --git a/src/mqt/debug/dap/messages/pause_dap_message.py b/src/mqt/debugger/dap/messages/pause_dap_message.py similarity index 100% rename from src/mqt/debug/dap/messages/pause_dap_message.py rename to src/mqt/debugger/dap/messages/pause_dap_message.py diff --git a/src/mqt/debug/dap/messages/restart_dap_message.py b/src/mqt/debugger/dap/messages/restart_dap_message.py similarity index 100% rename from src/mqt/debug/dap/messages/restart_dap_message.py rename to src/mqt/debugger/dap/messages/restart_dap_message.py diff --git a/src/mqt/debug/dap/messages/restart_frame_dap_message.py b/src/mqt/debugger/dap/messages/restart_frame_dap_message.py similarity index 100% rename from src/mqt/debug/dap/messages/restart_frame_dap_message.py rename to src/mqt/debugger/dap/messages/restart_frame_dap_message.py diff --git a/src/mqt/debug/dap/messages/reverse_continue_dap_message.py b/src/mqt/debugger/dap/messages/reverse_continue_dap_message.py similarity index 100% rename from src/mqt/debug/dap/messages/reverse_continue_dap_message.py rename to src/mqt/debugger/dap/messages/reverse_continue_dap_message.py diff --git a/src/mqt/debug/dap/messages/scopes_dap_message.py b/src/mqt/debugger/dap/messages/scopes_dap_message.py similarity index 100% rename from src/mqt/debug/dap/messages/scopes_dap_message.py rename to src/mqt/debugger/dap/messages/scopes_dap_message.py diff --git a/src/mqt/debug/dap/messages/set_breakpoints_dap_message.py b/src/mqt/debugger/dap/messages/set_breakpoints_dap_message.py similarity index 100% rename from src/mqt/debug/dap/messages/set_breakpoints_dap_message.py rename to src/mqt/debugger/dap/messages/set_breakpoints_dap_message.py diff --git a/src/mqt/debug/dap/messages/set_exception_breakpoints_dap_message.py b/src/mqt/debugger/dap/messages/set_exception_breakpoints_dap_message.py similarity index 100% rename from src/mqt/debug/dap/messages/set_exception_breakpoints_dap_message.py rename to src/mqt/debugger/dap/messages/set_exception_breakpoints_dap_message.py diff --git a/src/mqt/debug/dap/messages/stack_trace_dap_message.py b/src/mqt/debugger/dap/messages/stack_trace_dap_message.py similarity index 100% rename from src/mqt/debug/dap/messages/stack_trace_dap_message.py rename to src/mqt/debugger/dap/messages/stack_trace_dap_message.py diff --git a/src/mqt/debug/dap/messages/step_back_dap_message.py b/src/mqt/debugger/dap/messages/step_back_dap_message.py similarity index 100% rename from src/mqt/debug/dap/messages/step_back_dap_message.py rename to src/mqt/debugger/dap/messages/step_back_dap_message.py diff --git a/src/mqt/debug/dap/messages/step_in_dap_message.py b/src/mqt/debugger/dap/messages/step_in_dap_message.py similarity index 100% rename from src/mqt/debug/dap/messages/step_in_dap_message.py rename to src/mqt/debugger/dap/messages/step_in_dap_message.py diff --git a/src/mqt/debug/dap/messages/step_out_dap_message.py b/src/mqt/debugger/dap/messages/step_out_dap_message.py similarity index 100% rename from src/mqt/debug/dap/messages/step_out_dap_message.py rename to src/mqt/debugger/dap/messages/step_out_dap_message.py diff --git a/src/mqt/debug/dap/messages/stopped_dap_event.py b/src/mqt/debugger/dap/messages/stopped_dap_event.py similarity index 100% rename from src/mqt/debug/dap/messages/stopped_dap_event.py rename to src/mqt/debugger/dap/messages/stopped_dap_event.py diff --git a/src/mqt/debug/dap/messages/terminate_dap_message.py b/src/mqt/debugger/dap/messages/terminate_dap_message.py similarity index 91% rename from src/mqt/debug/dap/messages/terminate_dap_message.py rename to src/mqt/debugger/dap/messages/terminate_dap_message.py index e117fd6..0ea625c 100644 --- a/src/mqt/debug/dap/messages/terminate_dap_message.py +++ b/src/mqt/debugger/dap/messages/terminate_dap_message.py @@ -4,7 +4,7 @@ from typing import TYPE_CHECKING, Any -import mqt.debug +import mqt.debugger from .dap_message import DAPMessage @@ -37,5 +37,5 @@ def handle(self, server: DAPServer) -> dict[str, Any]: Returns: dict[str, Any]: The response to the request. """ - mqt.debug.destroy_ddsim_simulation_state(server.simulation_state) + mqt.debugger.destroy_ddsim_simulation_state(server.simulation_state) return super().handle(server) diff --git a/src/mqt/debug/dap/messages/terminated_dap_event.py b/src/mqt/debugger/dap/messages/terminated_dap_event.py similarity index 100% rename from src/mqt/debug/dap/messages/terminated_dap_event.py rename to src/mqt/debugger/dap/messages/terminated_dap_event.py diff --git a/src/mqt/debug/dap/messages/threads_dap_message.py b/src/mqt/debugger/dap/messages/threads_dap_message.py similarity index 100% rename from src/mqt/debug/dap/messages/threads_dap_message.py rename to src/mqt/debugger/dap/messages/threads_dap_message.py diff --git a/src/mqt/debug/dap/messages/utils.py b/src/mqt/debugger/dap/messages/utils.py similarity index 100% rename from src/mqt/debug/dap/messages/utils.py rename to src/mqt/debugger/dap/messages/utils.py diff --git a/src/mqt/debug/dap/messages/variables_dap_message.py b/src/mqt/debugger/dap/messages/variables_dap_message.py similarity index 100% rename from src/mqt/debug/dap/messages/variables_dap_message.py rename to src/mqt/debugger/dap/messages/variables_dap_message.py diff --git a/src/mqt/debugger/py.typed b/src/mqt/debugger/py.typed new file mode 100644 index 0000000..e69de29 diff --git a/src/mqt/debugger/pydebugger.pyi b/src/mqt/debugger/pydebugger.pyi new file mode 100644 index 0000000..26cd18e --- /dev/null +++ b/src/mqt/debugger/pydebugger.pyi @@ -0,0 +1,493 @@ +"""Type stubs for python bindings of the debug module.""" + +import enum + +# Enums + +class VariableType(enum.Enum): + """Represents possible types of classical variables.""" + + VarBool: VariableType + """A boolean variable.""" + VarInt: VariableType + """A 32-bit integer variable.""" + VarFloat: VariableType + """A floating-point variable.""" + +# Classes +class VariableValue: + """Represents the value of a classical variable. + + Only one of these fields has a valid value at a time, based on the variable's `VariableType`. + """ + + bool_value: bool + """The value of a boolean variable.""" + int_value: int + """The value of a 32-bit integer variable.""" + float_value: float + """The value of a floating-point variable.""" + + def __init__(self) -> None: + """Creates a new `VariableValue` instance.""" + +class Variable: + """Represents a classical variable.""" + + name: str + """The name of the variable.""" + type: VariableType + """The type of the variable.""" + value: VariableValue + """The value of the variable.""" + + def __init__(self) -> None: + """Creates a new `Variable` instance.""" + +class Complex: + """Represents a complex number.""" + + real: float + """The real part of the complex number.""" + imaginary: float + """The imaginary part of the complex number.""" + + def __init__(self, real: float = 0.0, imaginary: float = 0.0) -> None: + """Initializes a new complex number. + + Args: + real (float, optional): The real part of the complex number. Defaults to 0.0. + imaginary (float, optional): The imaginary part of the complex number. Defaults to 0.0. + """ + +class Statevector: + """Represents a state vector.""" + + num_qubits: int + """The number of qubits in the state vector.""" + num_states: int + """The number of states in the state vector. + + This is always equal to 2^`num_qubits`. + """ + + amplitudes: list[Complex] + """The amplitudes of the state vector. + + Contains one element for each of the `num_states` states in the state vector. + """ + + def __init__(self) -> None: + """Creates a new `Statevector` instance.""" + +class SimulationState: + """Represents the state of a quantum simulation for debugging. + + This is the main class of the `mqt-debugger` library, allowing developers to step through the code and inspect the state of the simulation. + """ + + def __init__(self) -> None: + """Creates a new `SimulationState` instance.""" + + def init(self) -> None: + """Initializes the simulation state.""" + + def load_code(self, code: str) -> None: + """Loads the given code into the simulation state. + + Args: + code (str): The code to load. + """ + + def step_forward(self) -> None: + """Steps the simulation forward by one instruction.""" + + def step_over_forward(self) -> None: + """Steps the simulation forward by one instruction, skipping over possible custom gate calls.""" + + def step_out_forward(self) -> None: + """Steps the simulation forward until the current custom gate call returns.""" + + def step_backward(self) -> None: + """Steps the simulation backward by one instruction.""" + + def step_over_backward(self) -> None: + """Steps the simulation backward by one instruction, skipping over possible custom gate calls.""" + + def step_out_backward(self) -> None: + """Steps the simulation backward until the instruction calling the current custom gate is encountered.""" + + def run_all(self) -> int: + """Runs the simulation until it finishes, even if assertions fail. + + Returns: + int: The number of assertions that failed during execution. + """ + + def run_simulation(self) -> None: + """Runs the simulation until it finishes or an assertion fails. + + If an assertion fails, the simulation stops and the `did_assertion_fail` + method will return `True`. + """ + + def run_simulation_backward(self) -> None: + """Runs the simulation backward until it finishes or an assertion fails.""" + + def reset_simulation(self) -> None: + """Resets the simulation to the initial state. + + This will reset measured variables and state vectors and go back to the + start of the code. + """ + + def pause_simulation(self) -> None: + """Pauses the simulation. + + If the simulation is running in a concurrent thread, the execution will + stop as soon as possible, but it is not guaranteed to stop immediately. + + If the simulation is not running, then the next call to continue the + simulation will stop as soon as possible. `step over` and `step out` + methods, in particular, may still execute the next instruction. + """ + + def can_step_forward(self) -> bool: + """Indicates whether the simulation can step forward. + + The simulation is unable to step forward if it has finished or if the + simulation has not been set up yet. + + Returns: + bool: True, if the simulation can step forward. + """ + + def can_step_backward(self) -> bool: + """Indicates whether the simulation can step backward. + + The simulation is unable to step backward if it is at the beginning or if + the simulation has not been set up yet. + + Returns: + bool: True, if the simulation can step backward. + """ + + def is_finished(self) -> bool: + """Indicates whether the execution has finished. + + The execution is considered finished if it has reached the end of the code. + + Returns: + bool: True, if the execution has finished. + """ + + def did_assertion_fail(self) -> bool: + """Indicates whether an assertion has failed in the previous step. + + If execution is continued after a failed assertion, then this flag will + be set to false again. + + Returns: + bool: True, if an assertion failed during the last step. + """ + + def was_breakpoint_hit(self) -> bool: + """Indicates whether a breakpoint was hit in the previous step. + + If execution is continued after a breakpoint, then this flag will + be set to false again. + + Returns: + bool: True, if a breakpoint was hit during the last step. + """ + + def get_current_instruction(self) -> int: + """Gets the current instruction index. + + Returns: + int: The index of the current instruction. + """ + + def get_instruction_count(self) -> int: + """Gets the number of instructions in the code. + + Returns: + int: The number of instructions in the code. + """ + + def get_instruction_position(self, instruction: int) -> tuple[int, int]: + """Gets the position of the given instruction in the code. + + Start and end positions are inclusive and white-spaces are ignored. + + Args: + instruction (int): The instruction to find. + + Returns: + tuple[int, int]: The start and end positions of the instruction. + """ + + def get_num_qubits(self) -> int: + """Gets the number of qubits used by the program. + + Returns: + int: The number of qubits used by the program. + """ + + def get_amplitude_index(self, index: int) -> Complex: + """Gets the complex amplitude of a state in the full state vector. + + The amplitude is selected by an integer index that corresponds to the + binary representation of the state. + + Args: + index (int): The index of the state in the full state vector. + + Returns: + Complex: The complex amplitude of the state. + """ + + def get_amplitude_bitstring(self, bitstring: str) -> Complex: + """Gets the complex amplitude of a state in the full state vector. + + The amplitude is selected by a bitstring representing the state. + + Args: + bitstring (str): The index of the state as a bitstring. + + Returns: + Complex: The complex amplitude of the state. + """ + + def get_classical_variable(self, name: str) -> Variable: + """Gets a classical variable by name. + + For registers, the name should be the register name followed by the index + in square brackets. + + Args: + name (str): The name of the variable. + + Returns: + Variable: The fetched variable. + """ + + def get_num_classical_variables(self) -> int: + """Gets the number of classical variables in the simulation. + + For registers, each index is counted as a separate variable. + + Returns: + int: The number of classical variables in the simulation. + """ + + def get_classical_variable_name(self, index: int) -> str: + """Gets the name of a classical variable by its index. + + For registers, each index is counted as a separate variable and can be + accessed separately. This method will return the name of the specific + index of the register. + + Args: + index (int): The index of the variable. + + Returns: + str: The name of the variable. + """ + + def get_state_vector_full(self) -> Statevector: + """Gets the full state vector of the simulation at the current time. + + The state vector is expected to be initialized with the correct number of + qubits and allocated space for the amplitudes before calling this method. + + Returns: + Statevector: The full state vector of the current simulation state. + """ + + def get_state_vector_sub(self, qubits: list[int]) -> Statevector: + """Gets a sub-state of the state vector of the simulation at the current time. + + The state vector is expected to be initialized with the correct number of + qubits and allocated space for the amplitudes before calling this method. + + This method also supports the re-ordering of qubits, but does not allow + qubits to be repeated. + + Args: + qubits (list[int]): The qubits to include in the sub-state. + + Returns: + Statevector: The sub-state vector of the current simulation state. + """ + + def set_breakpoint(self, desired_position: int) -> int: + """Sets a breakpoint at the desired position in the code. + + The position is given as a 0-indexed character position in the full code + string. + + Args: + desired_position (int): The position in the code to set the breakpoint. + + Returns: + int: The index of the instruction where the breakpoint was set. + """ + + def clear_breakpoints(self) -> None: + """Clears all breakpoints set in the simulation.""" + + def get_stack_depth(self) -> int: + """Gets the current stack depth of the simulation. + + Each custom gate call corresponds to one stack entry. + + Returns: + int: The current stack depth of the simulation. + """ + + def get_stack_trace(self, max_depth: int) -> list[int]: + """Gets the current stack trace of the simulation. + + The stack trace is represented as a list of instruction indices. Each + instruction index represents a single return address for the corresponding + stack entry. + + Args: + max_depth (int): The maximum depth of the stack trace. + + Returns: + list[int]: The stack trace of the simulation. + """ + + def get_diagnostics(self) -> Diagnostics: + """Gets the diagnostics instance employed by this debugger. + + Returns: + Diagnostics: The diagnostics instance employed by this debugger. + """ + +class ErrorCauseType(enum.Enum): + """Represents the type of a potential error cause.""" + + Unknown: ErrorCauseType + """An unknown error cause.""" + MissingInteraction: ErrorCauseType + """Indicates that an entanglement error may be caused by a missing interaction.""" + ControlAlwaysZero: ErrorCauseType + """Indicates that an error may be related to a controlled gate with a control that is always zero.""" + +class ErrorCause: + """Represents a potential cause of an assertion error.""" + + instruction: int + """The instruction where the error may originate from or where the error can be detected.""" + type: ErrorCauseType + """The type of the potential error cause.""" + + def __init__(self) -> None: + """Creates a new `ErrorCause` instance.""" + +class Diagnostics: + """Provides diagnostics capabilities such as different analysis methods for the debugger.""" + def __init__(self) -> None: + """Creates a new `Diagnostics` instance.""" + + def init(self) -> None: + """Initializes the diagnostics instance.""" + + def get_num_qubits(self) -> int: + """Get the number of qubits in the system. + + Returns: + int: The number of qubits in the system. + """ + + def get_instruction_count(self) -> int: + """Get the number of instructions in the code. + + Returns: + int: The number of instructions in the code. + """ + + def get_data_dependencies(self, instruction: int, include_callers: bool = False) -> list[int]: + """Extract all data dependencies for a given instruction. + + If the instruction is inside a custom gate definition, the data + dependencies will by default not go outside of the custom gate, unless a + new call instruction is found. By setting `include_callers` to `True`, all + possible callers of the custom gate will also be included and further + dependencies outside the custom gate will be taken from there. + + The line itself will also be counted as a dependency. Gate and register + declarations will not. + + This method can be performed without running the program, as it is a static + analysis method. + + Args: + instruction (int): The instruction to extract the data dependencies for. + include_callers (bool, optional): True, if the data dependencies should include all possible callers of the containing custom gate. Defaults to False. + + Returns: + list[int]: A list of instruction indices that are data dependencies of the given instruction. + """ + + def get_interactions(self, before_instruction: int, qubit: int) -> list[int]: + """Extract all qubits that interact with a given qubit up to a specific instruction. + + If the target instruction is inside a custom gate definition, the + interactions will only be searched inside the custom gate, unless a new + call instruction is found. + + The qubit itself will also be counted as an interaction. + + This method can be performed without running the program, as it is a static + analysis method. + + Args: + before_instruction (int): The instruction to extract the interactions up to (excluding). + qubit (int): The qubit to extract the interactions for. + + Returns: + list[int]: A list of qubit indices that interact with the given qubit up to the target instruction. + """ + + def get_zero_control_instructions(self) -> list[int]: + """Extract all controlled gates that have been marked as only having controls with value zero. + + This method expects a continuous memory block of booleans with size equal + to the number of instructions. Each element represents an instruction and + will be set to true if the instruction is a controlled gate with only zero + controls. + + This method can only be performed at runtime, as it is a dynamic analysis + method. + + Returns: + list[int]: The indices of instructions that are controlled gates with only zero controls. + """ + + def potential_error_causes(self) -> list[ErrorCause]: + """Extract a list of potential error causes encountered during execution. + + This method should be run after the program has been executed and reached + an assertion error. + + Returns: + list[ErrorCause]: A list of potential error causes encountered during execution. + """ + +def create_ddsim_simulation_state() -> SimulationState: + """Creates a new `SimulationState` instance using the DD backend for simulation and the OpenQASM language as input format. + + Returns: + SimulationState: The created simulation state. + """ + +def destroy_ddsim_simulation_state(state: SimulationState) -> None: + """Delete a given DD-based `SimulationState` instance and free up resources. + + Args: + state (SimulationState): The simulation state to delete. + """ diff --git a/src/python/CMakeLists.txt b/src/python/CMakeLists.txt index 5009707..fdf0404 100644 --- a/src/python/CMakeLists.txt +++ b/src/python/CMakeLists.txt @@ -1,5 +1,5 @@ pybind11_add_module( - pydebug + pydebugger # Prefer thin LTO if available THIN_LTO # Optimize the bindings for size @@ -8,11 +8,11 @@ pybind11_add_module( bindings.cpp InterfaceBindings.cpp dd/DDSimDebugBindings.cpp) -target_link_libraries(pydebug PRIVATE MQT::Debug MQT::CorePython pybind11_json MQT::ProjectOptions - MQT::ProjectWarnings) +target_link_libraries(pydebugger PRIVATE MQT::Debugger MQT::CorePython pybind11_json + MQT::ProjectOptions MQT::ProjectWarnings) message(STATUS "CMAKE_INSTALL_PREFIX is set to ${CMAKE_INSTALL_PREFIX}") # Install directive for scikit-build-core install( - TARGETS pydebug + TARGETS pydebugger DESTINATION . - COMPONENT mqt-debug_Python) + COMPONENT mqt-debugger_Python) diff --git a/src/python/InterfaceBindings.cpp b/src/python/InterfaceBindings.cpp index 95bb646..229e36b 100644 --- a/src/python/InterfaceBindings.cpp +++ b/src/python/InterfaceBindings.cpp @@ -47,235 +47,514 @@ struct StatevectorCPP { void bindFramework(py::module& m) { // Bind the VariableType enum py::enum_(m, "VariableType") - .value("VarBool", VarBool) - .value("VarInt", VarInt) - .value("VarFloat", VarFloat) + .value("VarBool", VarBool, "A boolean variable.") + .value("VarInt", VarInt, "An integer variable.") + .value("VarFloat", VarFloat, "A floating-point variable.") .export_values(); // Bind the VariableValue union py::class_(m, "VariableValue") .def(py::init<>()) - .def_readwrite("bool_value", &VariableValue::boolValue) - .def_readwrite("int_value", &VariableValue::intValue) - .def_readwrite("float_value", &VariableValue::floatValue); + .def_readwrite("bool_value", &VariableValue::boolValue, + "The value of a boolean variable.") + .def_readwrite("int_value", &VariableValue::intValue, + "The value of an integer variable.") + .def_readwrite("float_value", &VariableValue::floatValue, + "The value of a floating-point variable.") + .doc() = R"(Represents the value of a classical variable. + +Only one of these fields has a valid value at a time, based on the variable's `VariableType`.)"; // Bind the Variable struct py::class_(m, "Variable") - .def(py::init<>()) - .def_readwrite("name", &Variable::name) - .def_readwrite("type", &Variable::type) - .def_readwrite("value", &Variable::value); + .def(py::init<>(), "Creates a new `Variable` instance.") + .def_readwrite("name", &Variable::name, "The name of the variable.") + .def_readwrite("type", &Variable::type, "The type of the variable.") + .def_readwrite("value", &Variable::value, "The value of the variable.") + .doc() = "Represents a classical variable."; // Bind the Complex struct py::class_(m, "Complex") - .def(py::init<>()) - .def(py::init()) - .def_readwrite("real", &Complex::real) - .def_readwrite("imaginary", &Complex::imaginary) - .def("__str__", - [](const Complex& self) { - return std::to_string(self.real) + " + " + - std::to_string(self.imaginary) + "i"; - }) - .def("__repr__", [](const Complex& self) { - return "Complex(" + std::to_string(self.real) + ", " + - std::to_string(self.imaginary) + ")"; - }); + .def(py::init<>(), R"(Initializes a new complex number. + +Args: + real (float, optional): The real part of the complex number. Defaults to 0.0. + imaginary (float, optional): The imaginary part of the complex number. Defaults to 0.0.)") + .def(py::init(), R"(Initializes a new complex number. + +Args: + real (float, optional): The real part of the complex number. Defaults to 0.0. + imaginary (float, optional): The imaginary part of the complex number. Defaults to 0.0.)") + .def_readwrite("real", &Complex::real, + "The real part of the complex number.") + .def_readwrite("imaginary", &Complex::imaginary, + "The imaginary part of the complex number.") + .def( + "__str__", + [](const Complex& self) { + return std::to_string(self.real) + " + " + + std::to_string(self.imaginary) + "i"; + }, + R"(Returns a string representation of the complex number. +Returns: + str: The string representation of the complex number. +)") + .def( + "__repr__", + [](const Complex& self) { + return "Complex(" + std::to_string(self.real) + ", " + + std::to_string(self.imaginary) + ")"; + }, + R"(Returns a string representation of the complex number. + +Returns: + str: The string representation of the complex number.)") + .doc() = "Represents a complex number."; // Bind the Statevector struct py::class_(m, "Statevector") - .def(py::init<>()) - .def_readwrite("num_qubits", &StatevectorCPP::numQubits) - .def_readwrite("num_states", &StatevectorCPP::numStates) - .def_readwrite("amplitudes", &StatevectorCPP::amplitudes); + .def(py::init<>(), "Creates a new `Statevector` instance.") + .def_readwrite("num_qubits", &StatevectorCPP::numQubits, + "The number of qubits in the state vector.") + .def_readwrite("num_states", &StatevectorCPP::numStates, + R"(The number of states in the state vector. + +This is always equal to 2^`num_qubits`.)") + .def_readwrite("amplitudes", &StatevectorCPP::amplitudes, + R"(The amplitudes of the state vector. + +Contains one element for each of the `num_states` states in the state vector.)") + .doc() = "Represents a state vector."; py::class_(m, "SimulationState") - .def(py::init<>()) - .def("init", - [](SimulationState* self) { checkOrThrow(self->init(self)); }) - .def("load_code", - [](SimulationState* self, const char* code) { - checkOrThrow(self->loadCode(self, code)); - }) - .def("step_forward", - [](SimulationState* self) { checkOrThrow(self->stepForward(self)); }) - .def("step_over_forward", - [](SimulationState* self) { - checkOrThrow(self->stepOverForward(self)); - }) - .def("step_out_forward", - [](SimulationState* self) { - checkOrThrow(self->stepOutForward(self)); - }) + .def(py::init<>(), "Creates a new `SimulationState` instance.") + .def( + "init", [](SimulationState* self) { checkOrThrow(self->init(self)); }, + "Initializes the simulation state.") + .def( + "load_code", + [](SimulationState* self, const char* code) { + checkOrThrow(self->loadCode(self, code)); + }, + R"(Loads the given code into the simulation state. + +Args: + code (str): The code to load.)") + .def( + "step_forward", + [](SimulationState* self) { checkOrThrow(self->stepForward(self)); }, + "Steps the simulation forward by one instruction.") + .def( + "step_over_forward", + [](SimulationState* self) { + checkOrThrow(self->stepOverForward(self)); + }, + "Steps the simulation forward by one instruction, skipping over " + "possible custom gate calls.") + .def( + "step_out_forward", + [](SimulationState* self) { + checkOrThrow(self->stepOutForward(self)); + }, + "Steps the simulation forward until the current custom gate call " + "returns.") .def( "step_backward", - [](SimulationState* self) { checkOrThrow(self->stepBackward(self)); }) - .def("step_over_backward", - [](SimulationState* self) { - checkOrThrow(self->stepOverBackward(self)); - }) - .def("step_out_backward", - [](SimulationState* self) { - checkOrThrow(self->stepOutBackward(self)); - }) - .def("run_all", - [](SimulationState* self) { - size_t errors = 0; - checkOrThrow(self->runAll(self, &errors)); - return errors; - }) - .def("run_simulation", - [](SimulationState* self) { - checkOrThrow(self->runSimulation(self)); - }) - .def("run_simulation_backward", - [](SimulationState* self) { - checkOrThrow(self->runSimulationBackward(self)); - }) - .def("reset_simulation", - [](SimulationState* self) { - checkOrThrow(self->resetSimulation(self)); - }) - .def("pause_simulation", - [](SimulationState* self) { - checkOrThrow(self->pauseSimulation(self)); - }) - .def("can_step_forward", - [](SimulationState* self) { return self->canStepForward(self); }) - .def("can_step_backward", - [](SimulationState* self) { return self->canStepBackward(self); }) - .def("is_finished", - [](SimulationState* self) { return self->isFinished(self); }) - .def("did_assertion_fail", - [](SimulationState* self) { return self->didAssertionFail(self); }) - .def("was_breakpoint_hit", - [](SimulationState* self) { return self->wasBreakpointHit(self); }) - .def("get_current_instruction", - [](SimulationState* self) { - return self->getCurrentInstruction(self); - }) + [](SimulationState* self) { checkOrThrow(self->stepBackward(self)); }, + "Steps the simulation backward by one instruction.") + .def( + "step_over_backward", + [](SimulationState* self) { + checkOrThrow(self->stepOverBackward(self)); + }, + "Steps the simulation backward by one instruction, skipping over " + "possible custom gate calls.") + .def( + "step_out_backward", + [](SimulationState* self) { + checkOrThrow(self->stepOutBackward(self)); + }, + "Steps the simulation backward until the instruction calling the " + "current custom gate is encountered.") + .def( + "run_all", + [](SimulationState* self) { + size_t errors = 0; + checkOrThrow(self->runAll(self, &errors)); + return errors; + }, + R"(Runs the simulation until it finishes, even if assertions fail. + +Returns: +int: The number of assertions that failed during execution.)") + .def( + "run_simulation", + [](SimulationState* self) { + checkOrThrow(self->runSimulation(self)); + }, + R"(Runs the simulation until it finishes or an assertion fails. + +If an assertion fails, the simulation stops and the `did_assertion_fail` +method will return `True`.)") + .def( + "run_simulation_backward", + [](SimulationState* self) { + checkOrThrow(self->runSimulationBackward(self)); + }, + "Runs the simulation backward until it finishes or an assertion " + "fails.") + .def( + "reset_simulation", + [](SimulationState* self) { + checkOrThrow(self->resetSimulation(self)); + }, + R"(Resets the simulation to the initial state. + +This will reset measured variables and state vectors and go back to the +start of the code.)") + .def( + "pause_simulation", + [](SimulationState* self) { + checkOrThrow(self->pauseSimulation(self)); + }, + R"(Pauses the simulation. + +If the simulation is running in a concurrent thread, the execution will +stop as soon as possible, but it is not guaranteed to stop immediately. + +If the simulation is not running, then the next call to continue the +simulation will stop as soon as possible. `step over` and `step out` +methods, in particular, may still execute the next instruction.)") + .def( + "can_step_forward", + [](SimulationState* self) { return self->canStepForward(self); }, + R"(Indicates whether the simulation can step forward. + +The simulation is unable to step forward if it has finished or if the +simulation has not been set up yet. + +Returns: +bool: True, if the simulation can step forward.)") + .def( + "can_step_backward", + [](SimulationState* self) { return self->canStepBackward(self); }, + R"(Indicates whether the simulation can step backward. + +The simulation is unable to step backward if it is at the beginning or if +the simulation has not been set up yet. + +Returns: +bool: True, if the simulation can step backward.)") + .def( + "is_finished", + [](SimulationState* self) { return self->isFinished(self); }, + R"(Indicates whether the execution has finished. + +The execution is considered finished if it has reached the end of the code. + +Returns: +bool: True, if the execution has finished.)") + .def( + "did_assertion_fail", + [](SimulationState* self) { return self->didAssertionFail(self); }, + R"(Indicates whether an assertion has failed in the previous step. + +If execution is continued after a failed assertion, then this flag will +be set to false again. + +Returns: +bool: True, if an assertion failed during the last step.)") + .def( + "was_breakpoint_hit", + [](SimulationState* self) { return self->wasBreakpointHit(self); }, + R"(Indicates whether a breakpoint was hit in the previous step. + +If execution is continued after a breakpoint, then this flag will +be set to false again. + +Returns: +bool: True, if a breakpoint was hit during the last step.)") + .def( + "get_current_instruction", + [](SimulationState* self) { + return self->getCurrentInstruction(self); + }, + R"(Gets the current instruction index. + +Returns: +int: The index of the current instruction.)") .def( "get_instruction_count", - [](SimulationState* self) { return self->getInstructionCount(self); }) - .def("get_instruction_position", - [](SimulationState* self, size_t instruction) { - size_t start = 0; - size_t end = 0; - checkOrThrow( - self->getInstructionPosition(self, instruction, &start, &end)); - return std::make_pair(start, end); - }) - .def("get_num_qubits", - [](SimulationState* self) { return self->getNumQubits(self); }) - .def("get_amplitude_index", - [](SimulationState* self, size_t qubit) { - Complex output; - checkOrThrow(self->getAmplitudeIndex(self, qubit, &output)); - return output; - }) - .def("get_amplitude_bitstring", - [](SimulationState* self, const char* bitstring) { - Complex output; - checkOrThrow( - self->getAmplitudeBitstring(self, bitstring, &output)); - return output; - }) - .def("get_classical_variable", - [](SimulationState* self, const char* name) { - Variable output; - checkOrThrow(self->getClassicalVariable(self, name, &output)); - return output; - }) - .def("get_num_classical_variables", - [](SimulationState* self) { - return self->getNumClassicalVariables(self); - }) - .def("get_classical_variable_name", - [](SimulationState* self, size_t variableIndex) { - std::string output(255, '\0'); - checkOrThrow(self->getClassicalVariableName(self, variableIndex, - output.data())); - const std::size_t pos = output.find_first_of('\0'); - if (pos != std::string::npos) { - output = output.substr(0, pos); - } - return output; - }) - .def("get_state_vector_full", - [](SimulationState* self) { - const size_t numQubits = self->getNumQubits(self); - const std::vector amplitudes(1 << numQubits); - StatevectorCPP result{numQubits, 1ULL << numQubits, amplitudes}; - Statevector output{numQubits, result.numStates, - result.amplitudes.data()}; - checkOrThrow(self->getStateVectorFull(self, &output)); - return result; - }) - .def("get_state_vector_sub", - [](SimulationState* self, std::vector qubits) { - const size_t numQubits = qubits.size(); - const std::vector amplitudes(1 << numQubits); - StatevectorCPP result{numQubits, 1ULL << numQubits, amplitudes}; - Statevector output{numQubits, result.numStates, - result.amplitudes.data()}; - checkOrThrow(self->getStateVectorSub(self, numQubits, - qubits.data(), &output)); - return result; - }) - .def("set_breakpoint", - [](SimulationState* self, size_t desiredPosition) { - size_t actualPosition = 0; - checkOrThrow( - self->setBreakpoint(self, desiredPosition, &actualPosition)); - return actualPosition; - }) - .def("clear_breakpoints", - [](SimulationState* self) { - checkOrThrow(self->clearBreakpoints(self)); - }) - .def("get_stack_depth", - [](SimulationState* self) { - size_t depth = 0; - checkOrThrow(self->getStackDepth(self, &depth)); - return depth; - }) - .def("get_stack_trace", - [](SimulationState* self, size_t maxDepth) { - size_t trueSize = 0; - checkOrThrow(self->getStackDepth(self, &trueSize)); - const size_t stackSize = std::min(maxDepth, trueSize); - std::vector stackTrace(stackSize); - checkOrThrow( - self->getStackTrace(self, maxDepth, stackTrace.data())); - return stackTrace; - }) + [](SimulationState* self) { return self->getInstructionCount(self); }, + R"(Gets the number of instructions in the code. + +Returns: + int: The number of instructions in the code.)") + .def( + "get_instruction_position", + [](SimulationState* self, size_t instruction) { + size_t start = 0; + size_t end = 0; + checkOrThrow( + self->getInstructionPosition(self, instruction, &start, &end)); + return std::make_pair(start, end); + }, + R"(Gets the position of the given instruction in the code. + +Start and end positions are inclusive and white-spaces are ignored. + +Args: + instruction (int): The instruction to find. + +Returns: + tuple[int, int]: The start and end positions of the instruction.)") + .def( + "get_num_qubits", + [](SimulationState* self) { return self->getNumQubits(self); }, + R"(Gets the number of qubits used by the program. + +Returns: + int: The number of qubits used by the program.)") + .def( + "get_amplitude_index", + [](SimulationState* self, size_t qubit) { + Complex output; + checkOrThrow(self->getAmplitudeIndex(self, qubit, &output)); + return output; + }, + R"(Gets the complex amplitude of a state in the full state vector. + +The amplitude is selected by an integer index that corresponds to the +binary representation of the state. + +Args: + index (int): The index of the state in the full state vector. + +Returns: + Complex: The complex amplitude of the state.)") + .def( + "get_amplitude_bitstring", + [](SimulationState* self, const char* bitstring) { + Complex output; + checkOrThrow(self->getAmplitudeBitstring(self, bitstring, &output)); + return output; + }, + R"(Gets the complex amplitude of a state in the full state vector. + +The amplitude is selected by a bitstring representing the state. + +Args: + bitstring (str): The index of the state as a bitstring. + +Returns: + Complex: The complex amplitude of the state.)") + .def( + "get_classical_variable", + [](SimulationState* self, const char* name) { + Variable output; + checkOrThrow(self->getClassicalVariable(self, name, &output)); + return output; + }, + R"(Gets a classical variable by name. + +For registers, the name should be the register name followed by the index +in square brackets. + +Args: + name (str): The name of the variable. + +Returns: + Variable: The fetched variable.)") + .def( + "get_num_classical_variables", + [](SimulationState* self) { + return self->getNumClassicalVariables(self); + }, + R"(Gets the number of classical variables in the simulation. + +For registers, each index is counted as a separate variable. + +Returns: + int: The number of classical variables in the simulation.)") + .def( + "get_classical_variable_name", + [](SimulationState* self, size_t variableIndex) { + std::string output(255, '\0'); + checkOrThrow(self->getClassicalVariableName(self, variableIndex, + output.data())); + const std::size_t pos = output.find_first_of('\0'); + if (pos != std::string::npos) { + output = output.substr(0, pos); + } + return output; + }, + R"(Gets the name of a classical variable by its index. + +For registers, each index is counted as a separate variable and can be +accessed separately. This method will return the name of the specific +index of the register. + +Args: + index (int): The index of the variable. + +Returns: + str: The name of the variable.)") + .def( + "get_state_vector_full", + [](SimulationState* self) { + const size_t numQubits = self->getNumQubits(self); + const std::vector amplitudes(1 << numQubits); + StatevectorCPP result{numQubits, 1ULL << numQubits, amplitudes}; + Statevector output{numQubits, result.numStates, + result.amplitudes.data()}; + checkOrThrow(self->getStateVectorFull(self, &output)); + return result; + }, + R"(Gets the full state vector of the simulation at the current time. + +The state vector is expected to be initialized with the correct number of +qubits and allocated space for the amplitudes before calling this method. + +Returns: + Statevector: The full state vector of the current simulation state.)") + .def( + "get_state_vector_sub", + [](SimulationState* self, std::vector qubits) { + const size_t numQubits = qubits.size(); + const std::vector amplitudes(1 << numQubits); + StatevectorCPP result{numQubits, 1ULL << numQubits, amplitudes}; + Statevector output{numQubits, result.numStates, + result.amplitudes.data()}; + checkOrThrow(self->getStateVectorSub(self, numQubits, qubits.data(), + &output)); + return result; + }, + R"(Gets a sub-state of the state vector of the simulation at the current time. + +The state vector is expected to be initialized with the correct number of +qubits and allocated space for the amplitudes before calling this method. + +This method also supports the re-ordering of qubits, but does not allow +qubits to be repeated. + +Args: + qubits (list[int]): The qubits to include in the sub-state. + +Returns: + Statevector: The sub-state vector of the current simulation state.)") + .def( + "set_breakpoint", + [](SimulationState* self, size_t desiredPosition) { + size_t actualPosition = 0; + checkOrThrow( + self->setBreakpoint(self, desiredPosition, &actualPosition)); + return actualPosition; + }, + R"(Sets a breakpoint at the desired position in the code. + +The position is given as a 0-indexed character position in the full code +string. + +Args: + desired_position (int): The position in the code to set the breakpoint. + +Returns: + int: The index of the instruction where the breakpoint was set.)") + .def( + "clear_breakpoints", + [](SimulationState* self) { + checkOrThrow(self->clearBreakpoints(self)); + }, + "Clears all breakpoints set in the simulation.") + .def( + "get_stack_depth", + [](SimulationState* self) { + size_t depth = 0; + checkOrThrow(self->getStackDepth(self, &depth)); + return depth; + }, + R"(Gets the current stack depth of the simulation. + +Each custom gate call corresponds to one stack entry. + +Returns: + int: The current stack depth of the simulation.)") + .def( + "get_stack_trace", + [](SimulationState* self, size_t maxDepth) { + size_t trueSize = 0; + checkOrThrow(self->getStackDepth(self, &trueSize)); + const size_t stackSize = std::min(maxDepth, trueSize); + std::vector stackTrace(stackSize); + checkOrThrow( + self->getStackTrace(self, maxDepth, stackTrace.data())); + return stackTrace; + }, + R"(Gets the current stack trace of the simulation. + +The stack trace is represented as a list of instruction indices. Each +instruction index represents a single return address for the corresponding +stack entry. + +Args: + max_depth (int): The maximum depth of the stack trace. + +Returns: + list[int]: The stack trace of the simulation.)") .def( "get_diagnostics", [](SimulationState* self) { return self->getDiagnostics(self); }, - py::return_value_policy::reference_internal); + py::return_value_policy::reference_internal, + R"(Gets the diagnostics instance employed by this debugger. + +Returns: + Diagnostics: The diagnostics instance employed by this debugger.)") + .doc() = R"(Represents the state of a quantum simulation for debugging. + +"This is the main class of the `mqt-debugger` library, allowing developers to step through the code and inspect the state of the simulation.)"; } void bindDiagnostics(py::module& m) { // Bind the ErrorCauseType enum py::enum_(m, "ErrorCauseType") - .value("Unknown", Unknown) - .value("MissingInteraction", MissingInteraction) - .value("ControlAlwaysZero", ControlAlwaysZero) + .value("Unknown", Unknown, "An unknown error cause.") + .value("MissingInteraction", MissingInteraction, + "Indicates that an entanglement error may be caused by a missing " + "interaction.") + .value("ControlAlwaysZero", ControlAlwaysZero, + "Indicates that an error may be related to a controlled gate with " + "a control that is always zero.") .export_values(); // Bind the ErrorCause struct py::class_(m, "ErrorCause") .def(py::init<>()) - .def_readwrite("instruction", &ErrorCause ::instruction) - .def_readwrite("type", &ErrorCause ::type); + .def_readwrite("instruction", &ErrorCause ::instruction, + "The instruction where the error may originate from or " + "where the error can be detected.") + .def_readwrite("type", &ErrorCause ::type, + "The type of the potential error cause.") + .doc() = "Represents a potential cause of an assertion error."; py::class_(m, "Diagnostics") - .def(py::init<>()) - .def("init", [](Diagnostics* self) { checkOrThrow(self->init(self)); }) - .def("get_num_qubits", - [](Diagnostics* self) { return self->getNumQubits(self); }) - .def("get_instruction_count", - [](Diagnostics* self) { return self->getInstructionCount(self); }) + .def(py::init<>(), "Creates a new `Diagnostics` instance.") + .def( + "init", [](Diagnostics* self) { checkOrThrow(self->init(self)); }, + "Initializes the diagnostics instance.") + .def( + "get_num_qubits", + [](Diagnostics* self) { return self->getNumQubits(self); }, + R"(Get the number of qubits in the system. + +Returns: + int: The number of qubits in the system.)") + .def( + "get_instruction_count", + [](Diagnostics* self) { return self->getInstructionCount(self); }, + R"(Get the number of instructions in the code. + +Returns: + int: The number of instructions in the code.)") .def( "get_data_dependencies", [](Diagnostics* self, size_t instruction, bool includeCallers) { @@ -293,49 +572,111 @@ void bindDiagnostics(py::module& m) { } return result; }, - py::arg("instruction"), py::arg("include_callers") = false) - .def("get_interactions", - [](Diagnostics* self, size_t beforeInstruction, size_t qubit) { - std::vector qubits(self->getNumQubits(self)); - // NOLINTBEGIN(cppcoreguidelines-pro-type-reinterpret-cast) - checkOrThrow( - self->getInteractions(self, beforeInstruction, qubit, - reinterpret_cast(qubits.data()))); - // NOLINTEND(cppcoreguidelines-pro-type-reinterpret-cast) - std::vector result; - for (size_t i = 0; i < qubits.size(); i++) { - if (qubits[i] != 0) { - result.push_back(i); - } - } - return result; - }) - .def("get_zero_control_instructions", - [](Diagnostics* self) { - std::vector instructions(self->getInstructionCount(self)); - // NOLINTBEGIN(cppcoreguidelines-pro-type-reinterpret-cast) - checkOrThrow(self->getZeroControlInstructions( - self, reinterpret_cast(instructions.data()))); - // NOLINTEND(cppcoreguidelines-pro-type-reinterpret-cast) - std::vector result; - for (size_t i = 0; i < instructions.size(); i++) { - if (instructions[i] != 0) { - result.push_back(i); - } - } - return result; - }) - .def("potential_error_causes", [](Diagnostics* self) { - size_t nextSize = 10; - while (true) { - std::vector output(nextSize); - const auto actualSize = - self->potentialErrorCauses(self, output.data(), nextSize); - if (actualSize <= nextSize) { - output.resize(actualSize); - return output; - } - nextSize = nextSize * 2; - } - }); + py::arg("instruction"), py::arg("include_callers") = false, + R"(Extract all data dependencies for a given instruction. + +If the instruction is inside a custom gate definition, the data +dependencies will by default not go outside of the custom gate, unless a +new call instruction is found. By setting `include_callers` to `True`, all +possible callers of the custom gate will also be included and further +dependencies outside the custom gate will be taken from there. + +The line itself will also be counted as a dependency. Gate and register +declarations will not. + +This method can be performed without running the program, as it is a static +analysis method. + +Args: + instruction (int): The instruction to extract the data dependencies for. + include_callers (bool, optional): True, if the data dependencies should include all possible callers of the containing custom gate. Defaults to False. + +Returns: + list[int]: A list of instruction indices that are data dependencies of the given instruction.)") + .def( + "get_interactions", + [](Diagnostics* self, size_t beforeInstruction, size_t qubit) { + std::vector qubits(self->getNumQubits(self)); + // NOLINTBEGIN(cppcoreguidelines-pro-type-reinterpret-cast) + checkOrThrow( + self->getInteractions(self, beforeInstruction, qubit, + reinterpret_cast(qubits.data()))); + // NOLINTEND(cppcoreguidelines-pro-type-reinterpret-cast) + std::vector result; + for (size_t i = 0; i < qubits.size(); i++) { + if (qubits[i] != 0) { + result.push_back(i); + } + } + return result; + }, + R"(Extract all qubits that interact with a given qubit up to a specific instruction. + +If the target instruction is inside a custom gate definition, the +interactions will only be searched inside the custom gate, unless a new +call instruction is found. + +The qubit itself will also be counted as an interaction. + +This method can be performed without running the program, as it is a static +analysis method. + +Args: + before_instruction (int): The instruction to extract the interactions up to (excluding). + qubit (int): The qubit to extract the interactions for. + +Returns: + list[int]: A list of qubit indices that interact with the given qubit up to the target instruction.)") + .def( + "get_zero_control_instructions", + [](Diagnostics* self) { + std::vector instructions(self->getInstructionCount(self)); + // NOLINTBEGIN(cppcoreguidelines-pro-type-reinterpret-cast) + checkOrThrow(self->getZeroControlInstructions( + self, reinterpret_cast(instructions.data()))); + // NOLINTEND(cppcoreguidelines-pro-type-reinterpret-cast) + std::vector result; + for (size_t i = 0; i < instructions.size(); i++) { + if (instructions[i] != 0) { + result.push_back(i); + } + } + return result; + }, + R"(Extract all controlled gates that have been marked as only having controls with value zero. + +This method expects a continuous memory block of booleans with size equal +to the number of instructions. Each element represents an instruction and +will be set to true if the instruction is a controlled gate with only zero +controls. + +This method can only be performed at runtime, as it is a dynamic analysis +method. + +Returns: + list[int]: The indices of instructions that are controlled gates with only zero controls.)") + .def( + "potential_error_causes", + [](Diagnostics* self) { + size_t nextSize = 10; + while (true) { + std::vector output(nextSize); + const auto actualSize = + self->potentialErrorCauses(self, output.data(), nextSize); + if (actualSize <= nextSize) { + output.resize(actualSize); + return output; + } + nextSize = nextSize * 2; + } + }, + R"(Extract a list of potential error causes encountered during execution. + +This method should be run after the program has been executed and reached +an assertion error. + +Returns: + list[ErrorCause]: A list of potential error causes encountered during execution.)") + .doc() = "Provides diagnostics capabilities such as different analysis " + "methods for the debugger."; } diff --git a/src/python/bindings.cpp b/src/python/bindings.cpp index 6ce70de..eb162e2 100644 --- a/src/python/bindings.cpp +++ b/src/python/bindings.cpp @@ -1,6 +1,6 @@ /** * @file bindings.cpp - * @brief Python bindings for the debug module. + * @brief Python bindings for the debugger module. * * Central file for defining the Python bindings for the framework. */ @@ -10,7 +10,7 @@ #include -PYBIND11_MODULE(pydebug, m) { +PYBIND11_MODULE(pydebugger, m) { bindDiagnostics(m); bindFramework(m); bindBackend(m); diff --git a/src/python/dd/DDSimDebugBindings.cpp b/src/python/dd/DDSimDebugBindings.cpp index dae418a..e50ed43 100644 --- a/src/python/dd/DDSimDebugBindings.cpp +++ b/src/python/dd/DDSimDebugBindings.cpp @@ -14,16 +14,28 @@ void bindBackend(pybind11::module& m) { - m.def("create_ddsim_simulation_state", []() { - // NOLINTNEXTLINE(cppcoreguidelines-owning-memory) - auto* state = new DDSimulationState(); - createDDSimulationState(state); - return &state->interface; - }); + m.def( + "create_ddsim_simulation_state", + []() { + // NOLINTNEXTLINE(cppcoreguidelines-owning-memory) + auto* state = new DDSimulationState(); + createDDSimulationState(state); + return &state->interface; + }, + R"(Creates a new `SimulationState` instance using the DD backend for simulation and the OpenQASM language as input format. - m.def("destroy_ddsim_simulation_state", [](SimulationState* state) { - // NOLINTBEGIN(cppcoreguidelines-pro-type-reinterpret-cast) - destroyDDSimulationState(reinterpret_cast(state)); - // NOLINTEND(cppcoreguidelines-pro-type-reinterpret-cast) - }); +Returns: + SimulationState: The created simulation state.)"); + + m.def( + "destroy_ddsim_simulation_state", + [](SimulationState* state) { + // NOLINTBEGIN(cppcoreguidelines-pro-type-reinterpret-cast) + destroyDDSimulationState(reinterpret_cast(state)); + // NOLINTEND(cppcoreguidelines-pro-type-reinterpret-cast) + }, + R"(Delete a given DD-based `SimulationState` instance and free up resources. + +Args: + state (SimulationState): The simulation state to delete.)"); } diff --git a/test/CMakeLists.txt b/test/CMakeLists.txt index 0e33a05..3e0b574 100644 --- a/test/CMakeLists.txt +++ b/test/CMakeLists.txt @@ -1,6 +1,6 @@ package_add_test( - mqt_debug_test - MQT::Debug + mqt_debugger_test + MQT::Debugger utils_test.cpp test_utility.cpp test_simulation.cpp @@ -10,4 +10,4 @@ package_add_test( test_parsing.cpp) # set include directories -target_include_directories(mqt_debug_test PUBLIC ${PROJECT_SOURCE_DIR}/test/utils) +target_include_directories(mqt_debugger_test PUBLIC ${PROJECT_SOURCE_DIR}/test/utils) diff --git a/test/python/test_diagnosis.py b/test/python/test_diagnosis.py index b4aa97c..709aa6b 100644 --- a/test/python/test_diagnosis.py +++ b/test/python/test_diagnosis.py @@ -5,7 +5,7 @@ import locale from pathlib import Path -from mqt.debug import ( +from mqt.debugger import ( ErrorCauseType, SimulationState, create_ddsim_simulation_state, diff --git a/test/python/test_end_to_end.py b/test/python/test_end_to_end.py index 5993cd3..9ba30da 100644 --- a/test/python/test_end_to_end.py +++ b/test/python/test_end_to_end.py @@ -9,8 +9,8 @@ import pytest -from mqt.debug import Complex, SimulationState, Statevector, create_ddsim_simulation_state -from mqt.debug.pydebug import destroy_ddsim_simulation_state +from mqt.debugger import Complex, SimulationState, Statevector, create_ddsim_simulation_state +from mqt.debugger.pydebugger import destroy_ddsim_simulation_state @dataclass diff --git a/test/python/test_python_bindings.py b/test/python/test_python_bindings.py index 289960a..e0c860c 100644 --- a/test/python/test_python_bindings.py +++ b/test/python/test_python_bindings.py @@ -11,44 +11,44 @@ import pytest -import mqt.debug +import mqt.debugger -SimulationInstance = Tuple[mqt.debug.SimulationState, int] +SimulationInstance = Tuple[mqt.debugger.SimulationState, int] @pytest.fixture(scope="module") def simulation_instance_ghz() -> Generator[SimulationInstance, None, None]: """Fixture for the GHZ state simulation instance.""" - simulation_state = mqt.debug.create_ddsim_simulation_state() + simulation_state = mqt.debugger.create_ddsim_simulation_state() with Path("test/python/resources/bindings/ghz-incorrect.qasm").open( encoding=locale.getpreferredencoding(False) ) as f: code = f.read() simulation_state.load_code(code) yield (simulation_state, 0) - mqt.debug.destroy_ddsim_simulation_state(simulation_state) + mqt.debugger.destroy_ddsim_simulation_state(simulation_state) @pytest.fixture(scope="module") def simulation_instance_jumps() -> Generator[SimulationInstance, None, None]: """Fixture for the Jumps simulation instance.""" - simulation_state = mqt.debug.create_ddsim_simulation_state() + simulation_state = mqt.debugger.create_ddsim_simulation_state() with Path("test/python/resources/bindings/jumps.qasm").open(encoding=locale.getpreferredencoding(False)) as f: code = f.read() simulation_state.load_code(code) yield (simulation_state, 1) - mqt.debug.destroy_ddsim_simulation_state(simulation_state) + mqt.debugger.destroy_ddsim_simulation_state(simulation_state) @pytest.fixture(scope="module") def simulation_instance_classical() -> Generator[SimulationInstance, None, None]: """Fixture for the Classical simulation instance.""" - simulation_state = mqt.debug.create_ddsim_simulation_state() + simulation_state = mqt.debugger.create_ddsim_simulation_state() with Path("test/python/resources/bindings/classical.qasm").open(encoding=locale.getpreferredencoding(False)) as f: code = f.read() simulation_state.load_code(code) yield (simulation_state, 2) - mqt.debug.destroy_ddsim_simulation_state(simulation_state) + mqt.debugger.destroy_ddsim_simulation_state(simulation_state) @pytest.fixture(autouse=True) @@ -67,9 +67,9 @@ def simulation_state_cleanup( simulation_instance_classical[0].clear_breakpoints() -def load_fixture(request: pytest.FixtureRequest, name: str) -> tuple[mqt.debug.SimulationState, int]: +def load_fixture(request: pytest.FixtureRequest, name: str) -> tuple[mqt.debugger.SimulationState, int]: """Loads a fixture with the given name.""" - return cast(Tuple[mqt.debug.SimulationState, int], request.getfixturevalue(name)) + return cast(Tuple[mqt.debugger.SimulationState, int], request.getfixturevalue(name)) @pytest.mark.parametrize( @@ -234,9 +234,9 @@ def test_classical_get(simulation_instance_classical: SimulationInstance) -> Non assert simulation_state.get_classical_variable("c[1]").name == "c[1]" assert simulation_state.get_classical_variable("c[2]").name == "c[2]" - assert simulation_state.get_classical_variable("c[0]").type == mqt.debug.VariableType.VarBool - assert simulation_state.get_classical_variable("c[1]").type == mqt.debug.VariableType.VarBool - assert simulation_state.get_classical_variable("c[2]").type == mqt.debug.VariableType.VarBool + assert simulation_state.get_classical_variable("c[0]").type == mqt.debugger.VariableType.VarBool + assert simulation_state.get_classical_variable("c[1]").type == mqt.debugger.VariableType.VarBool + assert simulation_state.get_classical_variable("c[2]").type == mqt.debugger.VariableType.VarBool first = simulation_state.get_classical_variable("c[0]").value.bool_value assert simulation_state.get_classical_variable("c[1]").value.bool_value == first